tdesktop/Telegram/SourceFiles/statistics/view/chart_rulers_view.cpp

156 lines
4.0 KiB
C++

/*
This file is part of Telegram Desktop,
the official desktop application for the Telegram messaging service.
For license and copyright information please follow this link:
https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
*/
#include "statistics/view/chart_rulers_view.h"
#include "data/data_statistics_chart.h"
#include "statistics/chart_lines_filter_controller.h"
#include "statistics/statistics_common.h"
#include "styles/style_basic.h"
#include "styles/style_statistics.h"
namespace Statistic {
ChartRulersView::ChartRulersView() = default;
void ChartRulersView::setChartData(
const Data::StatisticalChart &chartData,
ChartViewType type,
std::shared_ptr<LinesFilterController> linesFilter) {
_rulers.clear();
_isDouble = (type == ChartViewType::DoubleLinear);
if (_isDouble && (chartData.lines.size() == 2)) {
_linesFilter = std::move(linesFilter);
_leftPen = QPen(chartData.lines.front().color);
_rightPen = QPen(chartData.lines.back().color);
_leftLineId = chartData.lines.front().id;
_rightLineId = chartData.lines.back().id;
const auto firstMax = chartData.lines.front().maxValue;
const auto secondMax = chartData.lines.back().maxValue;
if (firstMax > secondMax) {
_isLeftLineScaled = false;
_scaledLineRatio = firstMax / float64(secondMax);
} else {
_isLeftLineScaled = true;
_scaledLineRatio = secondMax / float64(firstMax);
}
}
}
void ChartRulersView::paintRulers(
QPainter &p,
const QRect &r) {
const auto alpha = p.opacity();
for (auto &ruler : _rulers) {
p.setOpacity(alpha * ruler.alpha * kRulerLineAlpha);
for (const auto &line : ruler.lines) {
const auto lineRect = QRect(
0,
r.y() + r.height() * line.relativeValue,
r.x() + r.width(),
st::lineWidth);
p.fillRect(lineRect, st::boxTextFg);
}
}
p.setOpacity(alpha);
}
void ChartRulersView::paintCaptionsToRulers(
QPainter &p,
const QRect &r) {
const auto offset = r.y() - st::statisticsChartRulerCaptionSkip;
p.setFont(st::statisticsDetailsBottomCaptionStyle.font);
const auto alpha = p.opacity();
for (auto &ruler : _rulers) {
const auto rulerAlpha = alpha * ruler.alpha;
p.setOpacity(rulerAlpha);
for (const auto &line : ruler.lines) {
const auto y = offset + r.height() * line.relativeValue;
const auto hasLinesFilter = _isDouble && _linesFilter;
if (hasLinesFilter) {
p.setPen(_leftPen);
p.setOpacity(rulerAlpha * _linesFilter->alpha(_leftLineId));
} else {
p.setPen(st::windowSubTextFg);
}
p.drawText(
0,
y,
(!_isDouble)
? line.caption
: _isLeftLineScaled
? line.scaledLineCaption
: line.caption);
if (hasLinesFilter) {
p.setOpacity(rulerAlpha * _linesFilter->alpha(_rightLineId));
p.setPen(_rightPen);
p.drawText(
r.width() - line.rightCaptionWidth,
y,
_isLeftLineScaled
? line.caption
: line.scaledLineCaption);
}
}
}
p.setOpacity(alpha);
}
void ChartRulersView::computeRelative(
int newMaxHeight,
int newMinHeight) {
for (auto &ruler : _rulers) {
ruler.computeRelative(newMaxHeight, newMinHeight);
}
}
void ChartRulersView::setAlpha(float64 value) {
for (auto &ruler : _rulers) {
ruler.alpha = ruler.fixedAlpha * (1. - value);
}
_rulers.back().alpha = value;
if (value == 1.) {
while (_rulers.size() > 1) {
const auto startIt = begin(_rulers);
if (!startIt->alpha) {
_rulers.erase(startIt);
} else {
break;
}
}
}
}
void ChartRulersView::add(Limits newHeight, bool animated) {
auto newLinesData = ChartRulersData(
newHeight.max,
newHeight.min,
true,
_isDouble ? _scaledLineRatio : 0.);
if (_isDouble) {
const auto &font = st::statisticsDetailsBottomCaptionStyle.font;
for (auto &line : newLinesData.lines) {
line.rightCaptionWidth = font->width(_isLeftLineScaled
? line.caption
: line.scaledLineCaption);
}
}
if (!animated) {
_rulers.clear();
}
for (auto &ruler : _rulers) {
ruler.fixedAlpha = ruler.alpha;
}
_rulers.push_back(newLinesData);
if (!animated) {
_rulers.back().alpha = 1.;
}
}
} // namespace Statistic