2011-07-01 10:10:44 +00:00
/*
2011-06-26 08:30:44 +00:00
gBarChart Implementation
Copyright ( c ) 2011 Mark Watkins < jedimark @ users . sourceforge . net >
License : GPL
2011-07-01 10:10:44 +00:00
*/
2011-06-26 08:30:44 +00:00
# include <math.h>
2011-09-03 12:59:08 +00:00
# include <QLabel>
2011-09-02 02:00:04 +00:00
# include <QDateTime>
2011-09-03 10:09:22 +00:00
# include "gYAxis.h"
2011-06-26 08:30:44 +00:00
# include "gBarChart.h"
2011-09-03 12:59:08 +00:00
extern QLabel * qstatus2 ;
2011-09-10 04:20:45 +00:00
SummaryChart : : SummaryChart ( Profile * p , QString label , GraphType type )
: Layer ( EmptyChannel ) , m_profile ( p ) , m_label ( label ) , m_graphtype ( type )
2011-06-26 08:30:44 +00:00
{
2011-09-10 04:20:45 +00:00
QColor color = Qt : : black ;
2011-09-02 05:13:07 +00:00
addGLBuf ( quads = new GLBuffer ( color , 20000 , GL_QUADS ) ) ;
2011-09-10 04:20:45 +00:00
addGLBuf ( lines = new GLBuffer ( color , 20000 , GL_LINES ) ) ;
2011-09-02 02:00:04 +00:00
quads - > forceAntiAlias ( true ) ;
2011-09-02 05:13:07 +00:00
m_empty = true ;
2011-09-03 10:09:22 +00:00
hl_day = - 1 ;
2011-06-26 08:30:44 +00:00
}
2011-09-10 04:20:45 +00:00
SummaryChart : : ~ SummaryChart ( )
2011-06-26 08:30:44 +00:00
{
}
2011-09-10 04:20:45 +00:00
void SummaryChart : : SetDay ( Day * nullday )
2011-09-02 05:13:07 +00:00
{
2011-09-10 04:20:45 +00:00
if ( ! m_profile ) {
qWarning ( ) < < " Forgot to set profile for gBarChart dummy! " ;
m_day = NULL ;
return ;
}
Day * day = nullday ;
Layer : : SetDay ( day ) ;
m_values . clear ( ) ;
m_miny = 9999999 ;
m_maxy = - 9999999 ;
m_minx = 0 ;
m_maxx = 0 ;
int dn ;
EventDataType tmp , total ;
ChannelID code ;
m_fday = 0 ;
qint64 tt ;
2011-09-03 04:38:12 +00:00
m_empty = true ;
2011-09-10 04:20:45 +00:00
SummaryType type ;
for ( QMap < QDate , QVector < Day * > > : : iterator d = m_profile - > daylist . begin ( ) ; d ! = m_profile - > daylist . end ( ) ; d + + ) {
tt = QDateTime ( d . key ( ) , QTime ( 0 , 0 , 0 ) , Qt : : UTC ) . toTime_t ( ) ;
//tt=QDateTime(d.key(),QTime(12,0,0)).toTime_t();
dn = tt / 86400 ;
tt * = 1000L ;
if ( ! m_minx | | tt < m_minx ) m_minx = tt ;
if ( ! m_maxx | | tt > m_maxx ) m_maxx = tt ;
total = 0 ;
bool fnd = false ;
for ( int j = 0 ; j < m_codes . size ( ) ; j + + ) {
code = m_codes [ j ] ;
type = m_type [ j ] ;
for ( int i = 0 ; i < d . value ( ) . size ( ) ; i + + ) {
day = d . value ( ) [ i ] ;
if ( type = = ST_HOURS | | day - > channelExists ( code ) ) { // too many lookups happening here.. stop the crap..
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 - > count ( code ) / day - > hours ( ) ; break ;
case ST_SPH : tmp = day - > sum ( code ) / day - > hours ( ) ; break ;
case ST_HOURS : tmp = day - > hours ( ) ; break ;
default : break ;
}
//if (tmp>0) {
fnd = true ;
total + = tmp ;
m_values [ dn ] [ j + 1 ] = tmp ;
2011-09-10 06:59:09 +00:00
if ( tmp < m_miny ) m_miny = tmp ;
if ( tmp > m_maxy ) m_maxy = tmp ;
2011-09-10 04:20:45 +00:00
break ;
// }
}
}
}
if ( fnd ) {
if ( ! m_fday ) m_fday = dn ;
m_values [ dn ] [ 0 ] = total ;
2011-09-10 06:59:09 +00:00
if ( m_graphtype = = GT_BAR ) {
if ( total < m_miny ) m_miny = total ;
if ( total > m_maxy ) m_maxy = total ;
}
2011-09-10 04:20:45 +00:00
m_empty = false ;
}
}
2011-09-10 06:59:09 +00:00
if ( m_graphtype = = GT_BAR ) {
m_miny = 0 ;
}
2011-09-10 04:20:45 +00:00
// m_minx=qint64(QDateTime(m_profile->FirstDay(),QTime(0,0,0),Qt::UTC).toTime_t())*1000L;
m_maxx = qint64 ( QDateTime ( m_profile - > LastDay ( ) . addDays ( 1 ) , QTime ( 0 , 0 , 0 ) , Qt : : UTC ) . toTime_t ( ) ) * 1000L ;
2011-09-02 05:13:07 +00:00
}
2011-06-26 08:30:44 +00:00
2011-09-10 04:20:45 +00:00
QColor brighten ( QColor color )
{
int cr , cg , cb ;
cr = color . red ( ) ;
cg = color . green ( ) ;
cb = color . blue ( ) ;
if ( cr < 64 ) cr = 64 ;
if ( cg < 64 ) cg = 64 ;
if ( cb < 64 ) cb = 64 ;
cr * = 2 ;
cg * = 2 ;
cb * = 2 ;
if ( cr > 255 ) 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 )
2011-06-26 08:30:44 +00:00
{
if ( ! m_visible ) return ;
2011-09-02 05:13:07 +00:00
2011-09-03 12:59:08 +00:00
rtop = top ;
2011-09-10 06:13:56 +00:00
GLBuffer * outlines = w . lines ( ) ;
2011-09-02 05:13:07 +00:00
QColor blk = Qt : : black ;
2011-09-10 06:13:56 +00:00
outlines - > add ( left , top , left , top + height , blk ) ;
outlines - > add ( left , top + height , left + width , top + height , blk ) ;
outlines - > add ( left + width , top + height , left + width , top , blk ) ;
outlines - > add ( left + width , top , left , top , blk ) ;
2011-09-02 02:00:04 +00:00
qint64 minx = w . min_x , maxx = w . max_x ;
2011-09-02 05:13:07 +00:00
//qint64 minx=m_minx, maxx=m_maxx;
2011-09-02 02:00:04 +00:00
qint64 xx = maxx - minx ;
2011-09-03 00:06:08 +00:00
float days = double ( xx ) / 86400000.0 ;
2011-09-02 05:13:07 +00:00
2011-09-03 02:11:10 +00:00
EventDataType maxy = m_maxy ;
EventDataType miny = m_miny ;
2011-09-03 02:51:55 +00:00
w . roundY ( miny , maxy ) ;
2011-09-03 02:11:10 +00:00
EventDataType yy = maxy - miny ;
EventDataType ymult = float ( height - 2 ) / yy ;
2011-09-02 02:00:04 +00:00
2011-09-03 10:09:22 +00:00
barw = ( float ( width ) / float ( days ) ) ;
2011-09-02 02:00:04 +00:00
2011-09-02 05:13:07 +00:00
qint64 ts ;
2011-09-02 02:00:04 +00:00
2011-09-03 10:09:22 +00:00
graph = & w ;
2011-09-02 05:13:07 +00:00
float px = left ;
2011-09-03 10:09:22 +00:00
l_left = w . m_marginleft + gYAxis : : Margin ;
l_top = w . m_margintop ;
l_width = width ;
l_height = height ;
2011-09-02 05:13:07 +00:00
float py ;
EventDataType total ;
int daynum = 0 ;
2011-09-10 04:20:45 +00:00
EventDataType h , tmp ;
2011-09-02 05:13:07 +00:00
2011-09-03 01:24:11 +00:00
2011-09-03 10:09:22 +00:00
l_offset = ( minx ) % 86400000L ;
offset = float ( l_offset ) / 86400000.0 ;
2011-09-03 00:06:08 +00:00
offset * = barw ;
px = left - offset ;
2011-09-03 10:09:22 +00:00
l_minx = minx ;
l_maxx = maxx + 86400000L ;
2011-09-03 00:06:08 +00:00
2011-09-10 04:20:45 +00:00
//QHash<short, EventDataType> lastvalues;
2011-09-03 02:11:10 +00:00
int total_days = 0 ;
double total_val = 0 ;
2011-09-10 04:20:45 +00:00
qint64 lastQ = 0 ;
bool lastdaygood = false ;
QVector < int > totalcounts ;
QVector < EventDataType > totalvalues ;
QVector < EventDataType > lastvalues ;
QVector < float > lastX ;
QVector < short > 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 < int , QHash < short , EventDataType > > : : iterator d = m_values . find ( zd ) ;
2011-09-10 06:59:09 +00:00
// if (d==m_values.end()) {
// d=m_values.find(zd--);
// }
2011-09-10 04:20:45 +00:00
lastdaygood = true ;
for ( int i = 0 ; i < numcodes ; i + + ) {
totalcounts [ i ] = 0 ;
totalvalues [ i ] = 0 ;
lastvalues [ i ] = 0 ;
lastX [ i ] = px ;
if ( d ! = m_values . end ( ) ) {
tmp = d . value ( ) [ i + 1 ] ;
h = tmp * ymult ;
} else {
lastdaygood = false ;
h = 0 ;
}
lastY [ i ] = top + height - 1 - h ;
}
2011-09-03 00:06:08 +00:00
for ( qint64 Q = minx ; Q < = maxx + 86400000L ; Q + = 86400000L ) {
2011-09-10 04:20:45 +00:00
zd = Q / 86400000L ;
2011-09-10 06:59:09 +00:00
d = m_values . find ( zd ) ;
2011-09-10 04:20:45 +00:00
qint64 extra = 86400000 ;
2011-09-03 00:06:08 +00:00
if ( Q < minx ) continue ;
2011-09-02 05:13:07 +00:00
if ( d ! = m_values . end ( ) ) {
2011-09-03 02:11:10 +00:00
int x1 = px , x2 = px + barw ;
if ( x1 < left ) x1 = left ;
if ( x2 > left + width ) x2 = left + width ;
if ( x2 < x1 ) continue ;
2011-09-03 00:06:08 +00:00
ChannelID code ;
2011-09-03 02:11:10 +00:00
2011-09-10 04:20:45 +00:00
total = d . value ( ) [ 0 ] ;
2011-09-03 02:11:10 +00:00
if ( total > 0 ) {
total_val + = total ;
total_days + + ;
}
2011-09-03 00:06:08 +00:00
py = top + height ;
2011-09-03 04:38:12 +00:00
for ( QHash < short , EventDataType > : : iterator g = d . value ( ) . begin ( ) ; g ! = d . value ( ) . end ( ) ; g + + ) {
short j = g . key ( ) ;
if ( ! j ) continue ;
2011-09-10 04:20:45 +00:00
j - - ;
QColor col = m_colors [ j ] ;
2011-09-03 10:09:22 +00:00
if ( zd = = hl_day ) {
col = QColor ( " gold " ) ;
}
2011-09-03 04:38:12 +00:00
2011-09-10 06:59:09 +00:00
tmp = g . value ( ) ;
2011-09-10 04:20:45 +00:00
totalvalues [ j ] + = tmp ;
2011-09-10 06:59:09 +00:00
totalcounts [ j ] + + ;
h = tmp * ymult ; // height in pixels
2011-09-10 04:20:45 +00:00
if ( m_graphtype = = GT_BAR ) {
QColor col2 = brighten ( col ) ;
quads - > add ( x1 , py , col ) ;
quads - > add ( x1 , py - h , col ) ;
quads - > add ( x2 , py - h , col2 ) ;
quads - > add ( x2 , py , col2 ) ;
if ( barw > 2 ) {
lines - > add ( x1 , py , x1 , py - h , blk ) ;
lines - > add ( x1 , py - h , x2 , py - h , blk ) ;
lines - > add ( x1 , py , x2 , py , blk ) ;
lines - > add ( x2 , py , x2 , py - h , blk ) ;
} // if (bar
py - = h ;
} else if ( m_graphtype = = GT_LINE ) { // if (m_graphtype==GT_BAR
2011-09-10 06:59:09 +00:00
short px2 = px + barw ;
2011-09-10 04:20:45 +00:00
short py2 = top + height - 1 - h ;
if ( lastdaygood ) {
lines - > add ( lastX [ j ] , lastY [ j ] , px2 , py2 , col ) ;
} else {
lines - > add ( x1 , py2 , x2 , py2 , col ) ;
}
lastX [ j ] = px2 ;
lastY [ j ] = py2 ;
//}
2011-09-05 02:30:10 +00:00
}
2011-09-10 04:20:45 +00:00
} // for(QHash<short
lastdaygood = true ;
if ( Q > maxx + extra ) break ;
} else lastdaygood = false ;
2011-09-02 05:13:07 +00:00
px + = barw ;
daynum + + ;
2011-09-10 04:20:45 +00:00
lastQ = Q ;
2011-09-02 02:00:04 +00:00
}
2011-09-10 04:20:45 +00:00
lines - > scissor ( left , w . flipY ( top + height + 2 ) , width + 1 , height + 1 ) ;
2011-09-03 02:11:10 +00:00
if ( total_days > 0 ) {
float val = total_val / float ( total_days ) ;
2011-09-07 08:35:55 +00:00
QString z = m_label + " = " + QString : : number ( val , ' f ' , 2 ) + " days= " + QString : : number ( total_days , ' f ' , 0 ) ;
2011-09-03 02:11:10 +00:00
w . renderText ( z , left , top - 1 ) ;
// val = AHI for selected area.
2011-06-26 08:30:44 +00:00
}
}
2011-09-10 04:20:45 +00:00
bool SummaryChart : : mouseMoveEvent ( QMouseEvent * event )
2011-09-03 10:09:22 +00:00
{
int x = event - > x ( ) - l_left ;
int y = event - > y ( ) - l_top ;
2011-09-04 05:41:55 +00:00
if ( ( x < 0 | | y < 0 | | x > l_width | | y > l_height ) ) {
2011-09-03 10:09:22 +00:00
hl_day = - 1 ;
2011-09-03 15:54:27 +00:00
//graph->timedRedraw(2000);
2011-09-03 10:09:22 +00:00
return false ;
}
double xx = l_maxx - l_minx ;
2011-09-10 04:20:45 +00:00
2011-09-03 10:09:22 +00:00
double xmult = xx / double ( l_width + barw ) ;
2011-09-05 13:26:10 +00:00
qint64 mx = ceil ( xmult * double ( x - offset ) ) ;
2011-09-03 10:09:22 +00:00
mx + = l_minx ;
mx = mx + l_offset ; //-86400000L;
2011-09-10 04:20:45 +00:00
// if (m_graphtype==GT_LINE) mx+=86400000L;
2011-09-03 10:09:22 +00:00
int zd = mx / 86400000L ;
2011-09-03 15:54:27 +00:00
//if (hl_day!=zd)
{
2011-09-03 10:09:22 +00:00
hl_day = zd ;
QHash < int , QHash < short , EventDataType > > : : iterator d = m_values . find ( hl_day ) ;
if ( d ! = m_values . end ( ) ) {
2011-09-03 12:59:08 +00:00
2011-09-05 10:28:41 +00:00
//int yy=y;
//int x=event->x()+graph->left+gGraphView::titleWidth;
;
//int x=event->x()+gYAxis::Margin-gGraphView::titleWidth;
//if (x>l_width-45) x=l_width-45;
x + = gYAxis : : Margin + gGraphView : : titleWidth ; //graph->m_marginleft+
int y = event - > y ( ) + rtop - 10 ;
/*int w=90;
2011-09-03 15:54:27 +00:00
int h = 32 ;
2011-09-04 11:29:12 +00:00
if ( x < 41 + w / 2 ) x = 41 + w / 2 ;
if ( y < 1 ) y = 1 ;
2011-09-05 10:28:41 +00:00
if ( y > l_height - h + 1 ) y = l_height - h + 1 ; */
2011-09-03 15:54:27 +00:00
2011-09-03 12:59:08 +00:00
2011-09-05 10:28:41 +00:00
//y+=rtop;
//TODO: Convert this to a ToolTip class
2011-09-03 15:54:27 +00:00
QDateTime dt = QDateTime : : fromTime_t ( hl_day * 86400 ) ;
2011-09-05 10:28:41 +00:00
QString z = dt . date ( ) . toString ( Qt : : SystemLocaleShortDate ) + " \n " + m_label + " = " + QString : : number ( d . value ( ) [ 0 ] , ' f ' , 2 ) ; ;
graph - > ToolTip ( z , x , y , 1500 ) ;
2011-09-03 12:59:08 +00:00
return true ;
}
//graph->redraw();
2011-09-03 10:09:22 +00:00
}
//qDebug() << l_left << x << hl_day << y << offset << barw;
return false ;
}
2011-09-10 04:20:45 +00:00
bool SummaryChart : : mousePressEvent ( QMouseEvent * event )
2011-09-03 10:09:22 +00:00
{
return false ;
}
2011-09-10 04:20:45 +00:00
bool SummaryChart : : mouseReleaseEvent ( QMouseEvent * event )
2011-09-03 10:09:22 +00:00
{
2011-09-04 05:41:55 +00:00
hl_day = - 1 ;
graph - > timedRedraw ( 2000 ) ;
2011-09-03 10:09:22 +00:00
return false ;
}
2011-09-10 04:20:45 +00:00
/*
2011-09-03 04:38:12 +00:00
UsageChart : : UsageChart ( Profile * profile )
: gBarChart ( )
{
m_profile = profile ;
m_colors . push_back ( Qt : : green ) ;
m_label = " Hours " ;
}
void UsageChart : : SetDay ( Day * day )
{
if ( ! m_profile ) {
qWarning ( ) < < " Forgot to set profile for gBarChart dummy! " ;
m_day = NULL ;
return ;
}
Layer : : SetDay ( day ) ;
//m_empty=true;
// if (!day) return;
m_values . clear ( ) ;
m_miny = 9999999 ;
m_maxy = - 9999999 ;
m_minx = 0 ;
m_maxx = 0 ;
int dn ;
EventDataType tmp , total ;
ChannelID code ;
m_fday = 0 ;
qint64 tt ;
2011-09-07 08:44:04 +00:00
int days = 0 ;
2011-09-03 04:38:12 +00:00
for ( QMap < QDate , QVector < Day * > > : : iterator d = m_profile - > daylist . begin ( ) ; d ! = m_profile - > daylist . end ( ) ; d + + ) {
tt = QDateTime ( d . key ( ) , QTime ( 0 , 0 , 0 ) , Qt : : UTC ) . toTime_t ( ) ;
dn = tt / 86400 ;
tt * = 1000L ;
2011-09-03 04:45:14 +00:00
// there could possibly may be a bug by doing this.. if charts don't match up.. come back here and enable the m_minx right down the bottom of this function.
2011-09-03 04:38:12 +00:00
if ( ! m_minx | | tt < m_minx ) m_minx = tt ;
if ( ! m_maxx | | tt > m_maxx ) m_maxx = tt ;
2011-09-03 04:45:14 +00:00
Day * day = m_profile - > GetDay ( d . key ( ) , MT_CPAP ) ;
if ( day ) {
total = day - > hours ( ) ;
2011-09-03 04:38:12 +00:00
2011-09-03 04:45:14 +00:00
m_values [ dn ] [ 0 ] = total ;
m_values [ dn ] [ 1 ] = total ;
if ( total < m_miny ) m_miny = total ;
if ( total > m_maxy ) m_maxy = total ;
2011-09-07 08:44:04 +00:00
days + + ;
2011-09-03 04:38:12 +00:00
}
}
2011-09-07 08:44:04 +00:00
if ( ! days ) m_empty = true ; else m_empty = false ;
2011-09-04 12:25:11 +00:00
//m_maxy=ceil(m_maxy);
2011-09-03 04:38:12 +00:00
//m_miny=floor(m_miny);
m_miny = 0 ;
// m_minx=qint64(QDateTime(m_profile->FirstDay(),QTime(0,0,0),Qt::UTC).toTime_t())*1000L;
m_maxx = qint64 ( QDateTime ( m_profile - > LastDay ( ) . addDays ( 1 ) , QTime ( 0 , 0 , 0 ) , Qt : : UTC ) . toTime_t ( ) ) * 1000L ;
}
AHIChart : : AHIChart ( Profile * profile )
: gBarChart ( )
{
m_label = " AHI " ;
m_profile = profile ;
}
2011-09-10 04:20:45 +00:00
void AHIChart : : SetDay ( Day * nullday )
2011-09-03 04:38:12 +00:00
{
2011-09-08 18:38:07 +00:00
}
AvgChart : : AvgChart ( Profile * profile )
: gBarChart ( )
{
m_label = " Avg " ;
m_profile = profile ;
}
void AvgChart : : SetDay ( Day * day )
{
if ( ! m_profile ) {
qWarning ( ) < < " Forgot to set profile for gBarChart dummy! " ;
m_day = NULL ;
return ;
}
Layer : : SetDay ( day ) ;
m_values . clear ( ) ;
m_miny = 9999999 ;
m_maxy = - 9999999 ;
m_minx = 0 ;
m_maxx = 0 ;
int dn ;
EventDataType tmp , total ;
ChannelID code ;
m_fday = 0 ;
qint64 tt ;
for ( QMap < QDate , QVector < Day * > > : : iterator d = m_profile - > daylist . begin ( ) ; d ! = m_profile - > daylist . end ( ) ; d + + ) {
tt = QDateTime ( d . key ( ) , QTime ( 0 , 0 , 0 ) , Qt : : UTC ) . toTime_t ( ) ;
//tt=QDateTime(d.key(),QTime(12,0,0)).toTime_t();
dn = tt / 86400 ;
tt * = 1000L ;
if ( ! m_minx | | tt < m_minx ) m_minx = tt ;
if ( ! m_maxx | | tt > m_maxx ) m_maxx = tt ;
total = 0 ;
bool fnd = false ;
for ( int j = 0 ; j < m_codes . size ( ) ; j + + ) {
code = m_codes [ j ] ;
for ( int i = 0 ; i < d . value ( ) . size ( ) ; i + + ) {
Day * day = d . value ( ) [ i ] ;
if ( day - > channelExists ( code ) ) { // too many lookups happening here.. stop the crap..
tmp = day - > wavg ( code ) ;
if ( tmp > 0 ) {
fnd = true ;
total + = tmp ;
m_values [ dn ] [ j + 1 ] = tmp ;
break ;
}
}
}
}
if ( fnd ) {
if ( ! m_fday ) m_fday = dn ;
m_values [ dn ] [ 0 ] = total ;
if ( total < m_miny ) m_miny = total ;
if ( total > m_maxy ) m_maxy = total ;
}
}
2011-09-03 04:38:12 +00:00
m_miny = 0 ;
// m_minx=qint64(QDateTime(m_profile->FirstDay(),QTime(0,0,0),Qt::UTC).toTime_t())*1000L;
m_maxx = qint64 ( QDateTime ( m_profile - > LastDay ( ) . addDays ( 1 ) , QTime ( 0 , 0 , 0 ) , Qt : : UTC ) . toTime_t ( ) ) * 1000L ;
m_empty = m_values . size ( ) = = 0 ;
}
2011-09-10 04:20:45 +00:00
*/