neobytes/src/qt/trafficgraphwidget.cpp

181 lines
5.4 KiB
C++

// Copyright (c) 2011-2015 The Bitcoin Core developers
// Distributed under the MIT software license, see the accompanying
// file COPYING or http://www.opensource.org/licenses/mit-license.php.
#include "trafficgraphwidget.h"
#include "clientmodel.h"
#include <boost/bind.hpp>
#include <QPainter>
#include <QColor>
#include <QTimer>
#include <cmath>
#define XMARGIN 10
#define YMARGIN 10
#define DEFAULT_SAMPLE_HEIGHT 1.1f
TrafficGraphWidget::TrafficGraphWidget(QWidget *parent) :
QWidget(parent),
timer(0),
fMax(DEFAULT_SAMPLE_HEIGHT),
nMins(0),
clientModel(0),
trafficGraphData(TrafficGraphData::Range_30m)
{
timer = new QTimer(this);
connect(timer, SIGNAL(timeout()), SLOT(updateRates()));
timer->setInterval(TrafficGraphData::SMALLEST_SAMPLE_PERIOD);
timer->start();
}
void TrafficGraphWidget::setClientModel(ClientModel *model)
{
clientModel = model;
if(model) {
trafficGraphData.setLastBytes(model->getTotalBytesRecv(), model->getTotalBytesSent());
}
}
int TrafficGraphWidget::getGraphRangeMins() const
{
return nMins;
}
void TrafficGraphWidget::paintPath(QPainterPath &path, const TrafficGraphData::SampleQueue &queue, SampleChooser chooser)
{
int h = height() - YMARGIN * 2, w = width() - XMARGIN * 2;
int sampleCount = queue.size(), x = XMARGIN + w, y;
if(sampleCount > 0) {
path.moveTo(x, YMARGIN + h);
for(int i = 0; i < sampleCount; ++i) {
x = XMARGIN + w - w * i / TrafficGraphData::DESIRED_DATA_SAMPLES;
y = YMARGIN + h - (int)(h * chooser(queue.at(i)) / fMax);
path.lineTo(x, y);
}
path.lineTo(x, YMARGIN + h);
}
}
namespace
{
float chooseIn(const TrafficSample& sample)
{
return sample.in;
}
float chooseOut(const TrafficSample& sample)
{
return sample.out;
}
}
void TrafficGraphWidget::paintEvent(QPaintEvent *)
{
QPainter painter(this);
painter.fillRect(rect(), Qt::black);
if(fMax <= 0.0f) return;
QColor axisCol(Qt::gray);
QColor axisCol2;
int h = height() - YMARGIN * 2;
painter.setPen(axisCol);
painter.drawLine(XMARGIN, YMARGIN + h, width() - XMARGIN, YMARGIN + h);
// decide what order of magnitude we are
int base = floor(log10(fMax));
float val = pow(10.0f, base);
float val2;
const QString units = tr("KB/s");
const float yMarginText = 2.0;
// draw lines
painter.setPen(axisCol);
for(float y = val; y < fMax; y += val) {
int yy = YMARGIN + h - h * y / fMax;
painter.drawLine(XMARGIN, yy, width() - XMARGIN, yy);
}
// if we drew 3 or fewer lines, break them up at the next lower order of magnitude
if(fMax / val <= 3.0f) {
axisCol2 = axisCol.darker();
val2 = pow(10.0f, base - 1);
painter.setPen(axisCol2);
int count = 1;
for(float y = val2; y < fMax; y += val2, count++) {
// don't overwrite lines drawn above
if(count % 10 == 0)
continue;
int yy = YMARGIN + h - h * y / fMax;
painter.drawLine(XMARGIN, yy, width() - XMARGIN, yy);
}
}
const TrafficGraphData::SampleQueue& queue = trafficGraphData.getCurrentRangeQueueWithAverageBandwidth();
if(!queue.empty()) {
QPainterPath pIn;
paintPath(pIn, queue, boost::bind(chooseIn,_1));
painter.fillPath(pIn, QColor(0, 255, 0, 128));
painter.setPen(Qt::green);
painter.drawPath(pIn);
QPainterPath pOut;
paintPath(pOut, queue, boost::bind(chooseOut,_1));
painter.fillPath(pOut, QColor(255, 0, 0, 128));
painter.setPen(Qt::red);
painter.drawPath(pOut);
}
// draw text on top of everything else
QRect textRect = painter.boundingRect(QRect(XMARGIN, YMARGIN + h - (h * val / fMax) - yMarginText, 0, 0), Qt::AlignLeft, QString("%1 %2").arg(val).arg(units));
textRect.translate(0, -textRect.height());
painter.fillRect(textRect, Qt::black);
painter.setPen(axisCol);
painter.drawText(textRect, Qt::AlignLeft, QString("%1 %2").arg(val).arg(units));
if(fMax / val <= 3.0f) {
QRect textRect2 = painter.boundingRect(QRect(XMARGIN, YMARGIN + h - (h * val2 / fMax) - yMarginText, 0, 0), Qt::AlignLeft, QString("%1 %2").arg(val2).arg(units));
textRect2.translate(0, -textRect2.height());
painter.fillRect(textRect2, Qt::black);
painter.setPen(axisCol2);
painter.drawText(textRect2, Qt::AlignLeft, QString("%1 %2").arg(val2).arg(units));
}
}
void TrafficGraphWidget::updateRates()
{
if(!clientModel) return;
bool updated = trafficGraphData.update(clientModel->getTotalBytesRecv(),clientModel->getTotalBytesSent());
if (updated){
float tmax = DEFAULT_SAMPLE_HEIGHT;
Q_FOREACH(const TrafficSample& sample, trafficGraphData.getCurrentRangeQueueWithAverageBandwidth()) {
if(sample.in > tmax) tmax = sample.in;
if(sample.out > tmax) tmax = sample.out;
}
fMax = tmax;
update();
}
}
void TrafficGraphWidget::setGraphRangeMins(int value)
{
trafficGraphData.switchRange(static_cast<TrafficGraphData::GraphRange>(value));
update();
}
void TrafficGraphWidget::clear()
{
trafficGraphData.clear();
fMax = DEFAULT_SAMPLE_HEIGHT;
if(clientModel) {
trafficGraphData.setLastBytes(clientModel->getTotalBytesRecv(), clientModel->getTotalBytesSent());
}
update();
}