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-02 02:00:04 +00:00
gBarChart : : gBarChart ( ChannelID code , QColor color , Qt : : Orientation o )
2011-08-30 08:46:24 +00:00
: Layer ( code ) , m_orientation ( o )
2011-06-26 08:30:44 +00:00
{
2011-09-02 05:13:07 +00:00
addGLBuf ( quads = new GLBuffer ( color , 20000 , GL_QUADS ) ) ;
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
}
gBarChart : : ~ gBarChart ( )
{
2011-09-02 02:00:04 +00:00
//delete Xaxis;
2011-06-26 08:30:44 +00:00
}
2011-09-02 05:13:07 +00:00
void gBarChart : : SetDay ( Day * day )
{
2011-09-03 04:38:12 +00:00
m_maxx = m_minx = 0 ;
m_maxy = m_miny = 0 ;
m_empty = true ;
2011-09-02 05:13:07 +00:00
}
2011-06-26 08:30:44 +00:00
2011-08-30 17:22:54 +00:00
void gBarChart : : 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
//if (!m_day) return;
2011-09-03 12:59:08 +00:00
rtop = top ;
2011-09-02 05:13:07 +00:00
GLBuffer * lines = w . lines ( ) ;
QColor blk = Qt : : black ;
lines - > add ( left , top , left , top + height , blk ) ;
lines - > add ( left , top + height , left + width , top + height , blk ) ;
lines - > add ( left + width , top + height , left + width , top , blk ) ;
lines - > 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 ;
float h , tmp ;
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-03 02:11:10 +00:00
int total_days = 0 ;
double total_val = 0 ;
2011-09-03 00:06:08 +00:00
for ( qint64 Q = minx ; Q < = maxx + 86400000L ; Q + = 86400000L ) {
2011-09-02 05:13:07 +00:00
int zd = Q / 86400000L ;
2011-09-03 04:38:12 +00:00
QHash < int , QHash < short , EventDataType > > : : iterator d = m_values . find ( zd ) ;
2011-09-03 00:06:08 +00:00
if ( Q < minx ) continue ;
if ( Q > maxx + 86400000 ) continue ; // break; // out of order if I end up using a hash instead.??
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 04:38:12 +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-03 10:09:22 +00:00
QColor col = m_colors [ j - 1 ] ;
2011-09-03 04:38:12 +00:00
int cr , cg , cb ;
cr = col . red ( ) ;
cg = col . green ( ) ;
cb = col . 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 ;
2011-09-03 10:09:22 +00:00
if ( zd = = hl_day ) {
col = QColor ( " gold " ) ;
}
2011-09-03 04:38:12 +00:00
QColor col2 = QColor ( cr , cg , cb , 255 ) ;
//col2=QColor(220,220,220,255);
tmp = g . value ( ) ; //(g.value()/float(total));
h = tmp * ymult ; //(float(total)*ymult); // height of chunk
quads - > add ( x1 , py , col ) ;
quads - > add ( x1 , py - h , col ) ;
quads - > add ( x2 , py - h , col2 ) ;
quads - > add ( x2 , py , col2 ) ;
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 ) ;
py - = h ;
2011-09-02 05:13:07 +00:00
}
}
px + = barw ;
daynum + + ;
2011-09-02 02:00:04 +00:00
}
2011-09-03 02:11:10 +00:00
if ( total_days > 0 ) {
float val = total_val / float ( total_days ) ;
2011-09-03 15:54:27 +00:00
QString z = m_label + " = " + QString : : number ( val , ' f ' , 2 ) + " days= " + QString : : number ( total_days , ' f ' , 0 ) + " This needs optimising and will going in overview " ;
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-03 10:09:22 +00:00
bool gBarChart : : mouseMoveEvent ( QMouseEvent * event )
{
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 ;
double xmult = xx / double ( l_width + barw ) ;
qint64 mx = xmult * double ( x - offset ) ;
mx + = l_minx ;
mx = mx + l_offset ; //-86400000L;
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
QColor col ( 255 , 255 , 128 , 200 ) ;
2011-09-03 15:54:27 +00:00
int yy = y ;
int x = event - > x ( ) - 10 ;
int w = 90 ;
int h = 32 ;
2011-09-04 11:29:12 +00:00
int y = event - > y ( ) - 42 ;
if ( x < 41 + w / 2 ) x = 41 + w / 2 ;
if ( y < 1 ) y = 1 ;
if ( x > 39 + l_width - w / 2 ) x = 39 + l_width - w / 2 ;
if ( y > l_height - h + 1 ) y = l_height - h + 1 ;
2011-09-03 15:54:27 +00:00
2011-09-04 11:29:12 +00:00
y + = rtop ;
2011-09-03 15:54:27 +00:00
//TODO: Convert this to a ToolTip class
graph - > quads ( ) - > add ( x , y , x , y + h , col ) ;
graph - > quads ( ) - > add ( x + w , y + h , x + w , y , col ) ;
2011-09-03 12:59:08 +00:00
QColor blk ( 0 , 0 , 0 , 255 ) ;
2011-09-03 15:54:27 +00:00
// The outer lines stuffs up
GLBuffer * lines = graph - > lines ( ) ; // toplines?
lines - > add ( x - 1 , y - 1 , x + w + 1 , y - 1 , blk ) ;
lines - > add ( x - 1 , y + h + 1 , x + w + 1 , y + h + 1 , blk ) ;
lines - > add ( x - 1 , y - 1 , x - 1 , y + h + 1 , blk ) ;
lines - > add ( x + w + 1 , y - 1 , x + w + 1 , y + h + 1 , blk ) ;
QDateTime dt = QDateTime : : fromTime_t ( hl_day * 86400 ) ;
QString z = dt . date ( ) . toString ( Qt : : SystemLocaleShortDate ) ;
2011-09-04 11:29:12 +00:00
graph - > renderText ( z , x + 10 , y + 11 ) ;
2011-09-03 15:54:27 +00:00
z = m_label + " = " + QString : : number ( d . value ( ) [ 0 ] , ' f ' , 2 ) ;
qstatus2 - > setText ( z ) ;
2011-09-04 11:29:12 +00:00
graph - > renderText ( z , x + 10 , y + 26 ) ;
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 ;
}
bool gBarChart : : mousePressEvent ( QMouseEvent * event )
{
return false ;
}
bool gBarChart : : mouseReleaseEvent ( QMouseEvent * event )
{
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-06-26 08:30:44 +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_empty = false ;
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 ( ) ;
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-03 04:38:12 +00:00
}
}
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 ;
}
void AHIChart : : 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 - > count ( code ) / day - > hours ( ) ;
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-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 ;
m_empty = m_values . size ( ) = = 0 ;
}