mirror of
https://gitlab.com/pholy/OSCAR-code.git
synced 2025-04-05 18:50:44 +00:00
295 lines
8.5 KiB
C++
295 lines
8.5 KiB
C++
/*
|
|
gSegmentChart Implementation
|
|
Copyright (c)2011 Mark Watkins <jedimark@users.sourceforge.net>
|
|
License: GPL
|
|
*/
|
|
|
|
#include <cmath>
|
|
#include "gSegmentChart.h"
|
|
|
|
|
|
gSegmentChart::gSegmentChart(GraphSegmentType type,QColor gradient_color,QColor outline_color)
|
|
:Layer(EmptyChannel),m_graph_type(type),m_gradient_color(gradient_color),m_outline_color(outline_color)
|
|
{
|
|
// m_gradient_color=QColor(200,200,200);
|
|
m_empty=true;
|
|
}
|
|
gSegmentChart::~gSegmentChart()
|
|
{
|
|
}
|
|
void gSegmentChart::AddSlice(ChannelID code,QColor color,QString name)
|
|
{
|
|
m_codes.push_back(code);
|
|
m_values.push_back(0);
|
|
m_colors.push_back(color);
|
|
m_names.push_back(name);
|
|
m_total=0;
|
|
}
|
|
void gSegmentChart::SetDay(Day *d)
|
|
{
|
|
Layer::SetDay(d);
|
|
m_total=0;
|
|
if (!m_day) return;
|
|
for (int c=0;c<m_codes.size();c++) {
|
|
m_values[c]=0;
|
|
for (QVector<Session *>::iterator s=m_day->begin();s!=m_day->end();s++) {
|
|
int cnt=(*s)->count(m_codes[c]);
|
|
m_values[c]+=cnt;
|
|
m_total+=cnt;
|
|
}
|
|
}
|
|
m_empty=true;
|
|
for (int i=0;i<m_codes.size();i++) {
|
|
if (m_day->count(m_codes[i])>0) {
|
|
m_empty=false;
|
|
break;
|
|
}
|
|
}
|
|
|
|
}
|
|
bool gSegmentChart::isEmpty()
|
|
{
|
|
return m_empty;
|
|
}
|
|
|
|
void gSegmentChart::paint(gGraph & w,int left, int top, int width, int height)
|
|
{
|
|
if (!m_visible) return;
|
|
if (!m_day) return;
|
|
//if (!m_total) return;
|
|
int start_px=left;
|
|
int start_py=top;
|
|
|
|
width--;
|
|
float diameter=MIN(width,height);
|
|
diameter-=8;
|
|
float radius=diameter/2.0;
|
|
|
|
float j=0.0;
|
|
float sum=0.0;
|
|
float step=1.0/45.0;
|
|
float px,py;
|
|
float q;
|
|
|
|
float xmult=float(width)/float(m_total);
|
|
float ymult=float(height)/float(m_total);
|
|
|
|
float xp=left;
|
|
|
|
int xoffset=width/2;
|
|
int yoffset=height/2;
|
|
if (m_total==0) {
|
|
QColor col=Qt::green;
|
|
QString a=":-)";
|
|
int x,y;
|
|
GetTextExtent(a,x,y,bigfont);
|
|
|
|
w.renderText(a,start_px+xoffset-x/2, (start_py+yoffset+y/2),0,col,bigfont);
|
|
return;
|
|
}
|
|
|
|
glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA);
|
|
glEnable(GL_BLEND);
|
|
glEnable(GL_LINE_SMOOTH);
|
|
glLineWidth(1.5);
|
|
int data;
|
|
unsigned size=m_values.size();
|
|
float line_step=float(width)/float(size-1);
|
|
bool line_first=true;
|
|
int line_last;
|
|
|
|
|
|
if (m_graph_type==GST_Line) {
|
|
w.qglColor(m_outline_color);
|
|
glBegin(GL_LINES);
|
|
}
|
|
|
|
for (unsigned m=0;m<size;m++) {
|
|
data=m_values[m];
|
|
|
|
/////////////////////////////////////////////////////////////////////////////////////
|
|
// Pie Chart
|
|
/////////////////////////////////////////////////////////////////////////////////////
|
|
if (m_graph_type==GST_Pie) {
|
|
j=float(data)/float(m_total); // ratio of this pie slice
|
|
|
|
// Draw Filling
|
|
glBegin(GL_POLYGON);
|
|
glPolygonMode(GL_BACK,GL_FILL);
|
|
w.qglColor(m_gradient_color);
|
|
glVertex2f(start_px+xoffset, start_py+height-yoffset);
|
|
w.qglColor(m_colors[m % m_colors.size()]);
|
|
for (q=sum;q<sum+j;q+=step) {
|
|
px=start_px+xoffset+sin(q*2*M_PI)*radius;
|
|
py=start_py+height-(yoffset+cos(q*2*M_PI)*radius);
|
|
glVertex2f(px,py);
|
|
}
|
|
q=sum+j;
|
|
px=start_px+xoffset+sin(q*2*M_PI)*radius;
|
|
py=start_py+height-(yoffset+cos(q*2*M_PI)*radius);
|
|
glVertex2f(px,py);
|
|
glEnd();
|
|
|
|
// Draw Outline
|
|
//m_outline_color=Qt::red;
|
|
w.qglColor(m_outline_color);
|
|
if (m_total>data) { // Draw the center point first
|
|
//glPolygonMode(GL_BACK,GL_LINE);
|
|
glBegin(GL_LINE_LOOP);
|
|
glVertex2f(start_px+xoffset, start_py+height-yoffset);
|
|
} else { // Only one entry, so just draw the circle
|
|
glBegin(GL_LINE_LOOP);
|
|
}
|
|
for (q=sum;q<sum+j;q+=step) {
|
|
px=start_px+xoffset+sin(q*2*M_PI)*radius;
|
|
py=start_py+height-(yoffset+cos(q*2*M_PI)*radius);
|
|
glVertex2f(px,py);
|
|
}
|
|
double tpx=start_px+xoffset+sin((sum+(j/2.0))*2*M_PI)*(radius/1.7);
|
|
double tpy=start_py+height-(yoffset+cos((sum+(j/2.0))*2*M_PI)*(radius/1.7));
|
|
q=sum+j;
|
|
px=start_px+xoffset+sin(q*2*M_PI)*radius;
|
|
py=start_py+height-(yoffset+cos(q*2*M_PI)*radius);
|
|
glVertex2f(px,py);
|
|
glEnd();
|
|
|
|
if (j>.09) {
|
|
//glBegin(GL_POINTS);
|
|
//glVertex2f(tpx,tpy);
|
|
//glEnd();
|
|
QString a=m_names[m]; //QString::number(floor(100.0/m_total*data),'f',0)+"%";
|
|
int x,y;
|
|
GetTextExtent(a,x,y);
|
|
w.renderText(a,tpx-(x/2.0),(tpy+y/2.0));
|
|
}
|
|
|
|
sum=q;
|
|
|
|
/////////////////////////////////////////////////////////////////////////////////////
|
|
// CandleStick Chart
|
|
/////////////////////////////////////////////////////////////////////////////////////
|
|
} else if (m_graph_type==GST_CandleStick) {
|
|
float bw=xmult*float(data);
|
|
|
|
glBegin(GL_QUADS);
|
|
w.qglColor(m_gradient_color);
|
|
glVertex2f(xp,start_py);
|
|
glVertex2f(xp+bw,start_py);
|
|
w.qglColor(m_colors[m % m_colors.size()]);
|
|
glVertex2f(xp+bw,start_py+height);
|
|
glVertex2f(xp,start_py+height);
|
|
glEnd();
|
|
|
|
w.qglColor(m_outline_color);
|
|
|
|
glBegin(GL_LINE_LOOP);
|
|
glVertex2f(xp,start_py);
|
|
glVertex2f(xp+bw,start_py);
|
|
glVertex2f(xp+bw,start_py+height);
|
|
glVertex2f(xp,start_py+height);
|
|
glEnd();
|
|
|
|
if (!m_names[m].isEmpty()) {
|
|
int px,py;
|
|
GetTextExtent(m_names[m],px,py);
|
|
if (px+5<bw) {
|
|
w.renderText(m_names[m],(xp+bw/2)-(px/2),top+((height/2)-(py/2)),0,Qt::black);
|
|
}
|
|
}
|
|
|
|
xp+=bw;
|
|
} else if (m_graph_type==GST_Line) {
|
|
float h=float(data)*ymult;
|
|
if (line_first) {
|
|
line_first=false;
|
|
} else {
|
|
glVertex2f(xp,line_last);
|
|
xp+=line_step;
|
|
glVertex2f(xp,h);
|
|
}
|
|
line_last=h;
|
|
}
|
|
}
|
|
if (m_graph_type==GST_Line) {
|
|
glEnd();
|
|
}
|
|
glPolygonMode(GL_BACK,GL_FILL);
|
|
glDisable(GL_LINE_SMOOTH);
|
|
glDisable(GL_BLEND);
|
|
}
|
|
|
|
|
|
gTAPGraph::gTAPGraph(ChannelID code,GraphSegmentType gt, QColor gradient_color,QColor outline_color)
|
|
:gSegmentChart(gt,gradient_color,outline_color),m_code(code)
|
|
{
|
|
m_colors.push_back(Qt::red);
|
|
m_colors.push_back(Qt::green);
|
|
}
|
|
gTAPGraph::~gTAPGraph()
|
|
{
|
|
}
|
|
void gTAPGraph::SetDay(Day *d)
|
|
{
|
|
Layer::SetDay(d);
|
|
m_total=0;
|
|
if (!m_day) return;
|
|
QMap<EventStoreType,qint64> tap;
|
|
|
|
EventStoreType data=0,lastval=0;
|
|
qint64 time=0,lasttime=0;
|
|
//bool first;
|
|
bool rfirst=true;
|
|
//bool changed;
|
|
EventDataType gain=1,offset=0;
|
|
for (QVector<Session *>::iterator s=m_day->begin();s!=m_day->end();s++) {
|
|
if ((*s)->eventlist.find(m_code)==(*s)->eventlist.end()) continue;
|
|
for (int q=0;q<(*s)->eventlist[m_code].size();q++) {
|
|
EventList &el=*(*s)->eventlist[m_code][q];
|
|
lasttime=el.time(0);
|
|
lastval=el.raw(0);
|
|
if (rfirst) {
|
|
gain=el.gain();
|
|
offset=el.offset();
|
|
rfirst=false;
|
|
}
|
|
//first=true;
|
|
//changed=false;
|
|
for (int i=1;i<el.count();i++) {
|
|
data=el.raw(i);
|
|
time=el.time(i);
|
|
if (lastval!=data) {
|
|
qint64 v=(time-lasttime);
|
|
if (tap.find(lastval)!=tap.end()) {
|
|
tap[lastval]+=v;
|
|
} else {
|
|
tap[lastval]=v;
|
|
}
|
|
//changed=true;
|
|
lasttime=time;
|
|
lastval=data;
|
|
}
|
|
}
|
|
if (time!=lasttime) {
|
|
qint64 v=(time-lasttime);
|
|
if (tap.find(lastval)!=tap.end()) {
|
|
tap[data]+=v;
|
|
} else {
|
|
tap[data]=v;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
m_values.clear();
|
|
m_names.clear();
|
|
m_total=0;
|
|
EventDataType val;
|
|
|
|
for (QMap<EventStoreType,qint64>::iterator i=tap.begin();i!=tap.end();i++) {
|
|
val=float(i.key())*gain+offset;
|
|
|
|
m_values.push_back(i.value()/1000L);
|
|
m_total+=i.value()/1000L;
|
|
m_names.push_back(QString::number(val,'f',2));
|
|
}
|
|
}
|