OSCAR-code/sleepyhead/Graphs/gSegmentChart.cpp

303 lines
9.1 KiB
C++
Raw Normal View History

/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 4 -*-
* vim: set ts=8 sts=4 et sw=4 tw=99:
*
* gSegmentChart Implementation
*
* Copyright (c) 2011-2014 Mark Watkins <jedimark@users.sourceforge.net>
*
* 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. */
2011-07-28 17:15:15 +00:00
#include <cmath>
#include "gSegmentChart.h"
gSegmentChart::gSegmentChart(GraphSegmentType type, QColor gradient_color, QColor outline_color)
: Layer(NoChannel), m_graph_type(type), m_gradient_color(gradient_color),
m_outline_color(outline_color)
2011-07-28 17:15:15 +00:00
{
m_empty = true;
2011-07-28 17:15:15 +00:00
}
gSegmentChart::~gSegmentChart()
{
}
void gSegmentChart::AddSlice(ChannelID code, QColor color, QString name)
2011-07-28 17:15:15 +00:00
{
m_codes.push_back(code);
m_values.push_back(0);
m_colors.push_back(color);
m_names.push_back(name);
m_total = 0;
2011-07-28 17:15:15 +00:00
}
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 (QList<Session *>::iterator s = m_day->begin(); s != m_day->end(); ++s) {
if ((*s)->enabled() && (*s)->m_cnt.contains(m_codes[c])) {
EventDataType cnt = (*s)->count(m_codes[c]);
m_values[c] += cnt;
m_total += cnt;
}
2011-07-28 17:15:15 +00:00
}
}
m_empty = true;
for (int i = 0; i < m_codes.size(); i++) {
if (m_day->count(m_codes[i]) > 0) {
m_empty = false;
break;
}
}
2011-08-30 08:46:24 +00:00
}
bool gSegmentChart::isEmpty()
{
return m_empty;
}
2011-07-28 17:15:15 +00:00
void gSegmentChart::paint(QPainter &painter, gGraph &w, const QRegion &region)
2011-07-28 17:15:15 +00:00
{
int left = region.boundingRect().left();
int top = region.boundingRect().top();
int width = region.boundingRect().width();
int height = region.boundingRect().height();
if (!m_visible) { return; }
if (!m_day) { return; }
int start_px = left;
int start_py = top;
2011-07-28 18:02:51 +00:00
width--;
float diameter = MIN(width, height);
diameter -= 8;
float radius = diameter / 2.0;
float xmult = float(width) / float(m_total);
float ymult = float(height) / float(m_total);
2011-07-28 17:15:15 +00:00
float xp = left;
2011-07-28 17:15:15 +00:00
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);
2011-08-07 07:46:59 +00:00
return;
}
2011-07-28 17:15:15 +00:00
EventDataType data;
unsigned size = m_values.size();
float line_step = float(width) / float(size - 1);
bool line_first = true;
2011-07-29 14:58:44 +00:00
int line_last;
2014-05-07 19:52:59 +00:00
float sum = -90.0;
2014-05-07 19:52:59 +00:00
painter.setFont(*defaultfont);
for (unsigned m = 0; m < size; m++) {
data = m_values[m];
2011-07-28 17:15:15 +00:00
if (data == 0) { continue; }
/////////////////////////////////////////////////////////////////////////////////////
// Pie Chart
/////////////////////////////////////////////////////////////////////////////////////
if (m_graph_type == GST_Pie) {
2014-08-05 11:17:03 +00:00
const QColor col = schema::channel[m_codes[m % m_colors.size()]].defaultColor();
2011-07-28 17:15:15 +00:00
2014-05-07 19:52:59 +00:00
// length of this segment in degrees
float len = 360.0 / float(m_total) * float(data);
2014-05-07 19:52:59 +00:00
// Setup the shiny radial gradient
2011-07-28 17:15:15 +00:00
2014-05-07 19:52:59 +00:00
painter.setRenderHint(QPainter::Antialiasing);
QRect pierect(start_px+1, start_py+1, width-2, height-2);
2014-05-07 19:52:59 +00:00
painter.setPen(QPen(col, 0));
QRadialGradient gradient(pierect.center(), pierect.width()/2, pierect.center());
gradient.setColorAt(0, Qt::white);
gradient.setColorAt(1, col);
// draw filled pie
painter.setBrush(gradient);
painter.setBackgroundMode(Qt::OpaqueMode);
painter.drawPie(pierect, -sum * 16.0, -len * 16.0);
// draw outline
painter.setBackgroundMode(Qt::TransparentMode);
painter.setBrush(QBrush(col,Qt::NoBrush));
painter.setPen(QPen(QColor(Qt::black),1.5));
painter.drawPie(pierect, -sum * 16.0, -len * 16.0);
2014-05-08 01:04:37 +00:00
// Draw text labels if they fit
if (len > 20) {
float angle = (sum+90.0) + len / 2.0;
double tpx = (pierect.x() + pierect.width()/2) + (sin((180 - angle) * (M_PI / 180.0)) * (radius / 1.65));
double tpy = (pierect.y() + pierect.height()/2) + (cos((180 - angle) * (M_PI / 180.0)) * (radius / 1.65));
2014-05-07 19:52:59 +00:00
QString txt = m_names[m]; //QString::number(floor(100.0/m_total*data),'f',0)+"%";
int x, y;
2014-05-07 19:52:59 +00:00
GetTextExtent(txt, x, y);
// antialiasing looks like crap here..
painter.setPen(QColor(Qt::black));
2014-05-08 01:04:37 +00:00
painter.drawText(tpx - (x / 2.0), tpy+3, txt);
2011-08-07 07:11:50 +00:00
}
2014-05-07 19:52:59 +00:00
sum += len;
2011-08-07 07:11:50 +00:00
2014-05-07 19:52:59 +00:00
} else if (m_graph_type == GST_CandleStick) {
/////////////////////////////////////////////////////////////////////////////////////
// CandleStick Chart
/////////////////////////////////////////////////////////////////////////////////////
2014-05-07 19:52:59 +00:00
QColor &col = m_colors[m % m_colors.size()];
2014-05-07 19:52:59 +00:00
float bw = xmult * float(data);
2011-07-28 17:15:15 +00:00
2014-05-07 19:52:59 +00:00
QLinearGradient linearGrad(QPointF(0, 0), QPointF(bw, 0));
linearGrad.setColorAt(0, col);
linearGrad.setColorAt(1, Qt::white);
painter.fillRect(xp, start_py, bw, height, QBrush(linearGrad));
2011-07-28 17:15:15 +00:00
2014-05-07 19:52:59 +00:00
painter.setPen(m_outline_color);
painter.drawLine(xp, start_py, xp + bw, start_py);
painter.drawLine(xp + bw, start_py + height, xp, start_py + height);
2011-07-28 17:15:15 +00:00
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);
2011-07-28 17:15:15 +00:00
}
}
xp += bw;
/////////////////////////////////////////////////////////////////////////////////////
// Line Chart
/////////////////////////////////////////////////////////////////////////////////////
} else if (m_graph_type == GST_Line) {
QColor col = Qt::black; //m_colors[m % m_colors.size()];
2014-05-07 19:52:59 +00:00
painter.setPen(col);
float h = (top + height) - (float(data) * ymult);
2011-07-29 14:58:44 +00:00
if (line_first) {
line_first = false;
2011-07-29 14:58:44 +00:00
} else {
2014-05-07 19:52:59 +00:00
painter.drawLine(xp, line_last, xp + line_step, h);
xp += line_step;
2011-07-29 14:58:44 +00:00
}
line_last = h;
2011-07-28 17:15:15 +00:00
}
}
}
gTAPGraph::gTAPGraph(ChannelID code, GraphSegmentType gt, QColor gradient_color,
QColor outline_color)
: gSegmentChart(gt, gradient_color, outline_color), m_code(code)
2011-07-28 17:15:15 +00:00
{
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;
2011-07-28 17:15:15 +00:00
EventStoreType data = 0, lastval = 0;
qint64 time = 0, lasttime = 0;
2011-08-31 05:24:48 +00:00
//bool first;
bool rfirst = true;
2011-08-31 05:24:48 +00:00
//bool changed;
EventDataType gain = 1, offset = 0;
QHash<ChannelID, QVector<EventList *> >::iterator ei;
for (QList<Session *>::iterator s = m_day->begin(); s != m_day->end(); ++s) {
if (!(*s)->enabled()) { continue; }
if ((ei = (*s)->eventlist.find(m_code)) == (*s)->eventlist.end()) { continue; }
for (int q = 0; q < ei.value().size(); q++) {
EventList &el = *(ei.value()[q]);
lasttime = el.time(0);
lastval = el.raw(0);
2011-08-07 11:37:56 +00:00
if (rfirst) {
gain = el.gain();
offset = el.offset();
rfirst = false;
2011-08-07 11:37:56 +00:00
}
2011-08-31 05:24:48 +00:00
//first=true;
//changed=false;
for (quint32 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;
2011-08-07 11:37:56 +00:00
} else {
tap[lastval] = v;
2011-07-28 17:15:15 +00:00
}
2011-08-31 05:24:48 +00:00
//changed=true;
lasttime = time;
lastval = data;
2011-07-28 17:15:15 +00:00
}
}
if (time != lasttime) {
qint64 v = (time - lasttime);
if (tap.find(lastval) != tap.end()) {
tap[data] += v;
2011-08-07 11:37:56 +00:00
} else {
tap[data] = v;
2011-08-07 11:37:56 +00:00
}
}
2011-07-28 17:15:15 +00:00
}
}
2011-07-28 17:15:15 +00:00
m_values.clear();
m_names.clear();
m_total = 0;
2011-07-28 17:15:15 +00:00
EventDataType val;
for (QMap<EventStoreType, qint64>::iterator i = tap.begin(); i != tap.end(); i++) {
val = float(i.key()) * gain + offset;
2011-08-07 11:37:56 +00:00
m_values.push_back(i.value() / 1000L);
m_total += i.value() / 1000L;
m_names.push_back(QString::number(val, 'f', 2));
2011-07-28 17:15:15 +00:00
}
m_empty = m_values.size() == 0;
2011-07-28 17:15:15 +00:00
}