mirror of
https://github.com/subsurface/subsurface.git
synced 2025-01-20 06:45:27 +00:00
405e6b6b69
The coordinates of these were calculated when creating the feature. This is wrong, because the min/max values of the axes can change on resize to get "nice" number. Therefore, recalculate after resizing. This means that the general "LineMarker" class has to be split into two classes, one for regression lines and one for median/mean markers. Signed-off-by: Berthold Stoeger <bstoeger@mail.tuwien.ac.at>
1089 lines
36 KiB
C++
1089 lines
36 KiB
C++
// SPDX-License-Identifier: GPL-2.0
|
|
#include "statsview.h"
|
|
#include "barseries.h"
|
|
#include "boxseries.h"
|
|
#include "legend.h"
|
|
#include "pieseries.h"
|
|
#include "scatterseries.h"
|
|
#include "statsaxis.h"
|
|
#include "statsgrid.h"
|
|
#include "statsstate.h"
|
|
#include "statstranslations.h"
|
|
#include "statsvariables.h"
|
|
#include "zvalues.h"
|
|
#include "core/divefilter.h"
|
|
#include "core/subsurface-qt/divelistnotifier.h"
|
|
|
|
#include <cmath>
|
|
#include <QQuickItem>
|
|
#include <QAbstractSeries>
|
|
#include <QChart>
|
|
#include <QGraphicsSceneHoverEvent>
|
|
#include <QGraphicsSimpleTextItem>
|
|
#include <QLocale>
|
|
|
|
// Constants that control the graph layouts
|
|
static const QColor quartileMarkerColor(Qt::red);
|
|
static const double quartileMarkerSize = 15.0;
|
|
static const double sceneBorder = 5.0; // Border between scene edges and statitistics view
|
|
static const double titleBorder = 2.0; // Border between title and chart
|
|
|
|
static const QUrl urlStatsView = QUrl(QStringLiteral("qrc:/qml/statsview.qml"));
|
|
|
|
// We use QtQuick's ChartView so that we can show the statistics on mobile.
|
|
// However, accessing the ChartView from C++ is maliciously cumbersome and
|
|
// the full QChart interface is not exported. Fortunately, the interface
|
|
// leaks the QChart object: We can create a dummy-series and access the chart
|
|
// object via the chart() accessor function. By creating a "PieSeries", the
|
|
// ChartView does not automatically add axes.
|
|
static QtCharts::QChart *getChart(QQuickItem *item)
|
|
{
|
|
QtCharts::QAbstractSeries *abstract_series;
|
|
if (!item)
|
|
return nullptr;
|
|
if (!QMetaObject::invokeMethod(item, "createSeries", Qt::AutoConnection,
|
|
Q_RETURN_ARG(QtCharts::QAbstractSeries *, abstract_series),
|
|
Q_ARG(int, QtCharts::QAbstractSeries::SeriesTypePie),
|
|
Q_ARG(QString, QString()))) {
|
|
qWarning("Couldn't call createSeries()");
|
|
return nullptr;
|
|
}
|
|
QtCharts::QChart *res = abstract_series->chart();
|
|
res->removeSeries(abstract_series);
|
|
delete abstract_series;
|
|
return res;
|
|
}
|
|
|
|
bool StatsView::EventFilter::eventFilter(QObject *o, QEvent *event)
|
|
{
|
|
if (event->type() == QEvent::GraphicsSceneHoverMove) {
|
|
QGraphicsSceneHoverEvent *hover = static_cast<QGraphicsSceneHoverEvent *>(event);
|
|
view->hover(hover->pos());
|
|
return true;
|
|
}
|
|
return QObject::eventFilter(o, event);
|
|
}
|
|
|
|
StatsView::StatsView(QWidget *parent) : QQuickWidget(parent),
|
|
highlightedSeries(nullptr),
|
|
xAxis(nullptr),
|
|
yAxis(nullptr),
|
|
eventFilter(this)
|
|
{
|
|
setResizeMode(QQuickWidget::SizeRootObjectToView);
|
|
// if we get a failure to load the QML file (e.g., when the QtCharts QML modules aren't found)
|
|
// the chart will be null
|
|
setSource(urlStatsView);
|
|
chart = getChart(rootObject());
|
|
if (chart) {
|
|
connect(chart, &QtCharts::QChart::plotAreaChanged, this, &StatsView::plotAreaChanged);
|
|
connect(&diveListNotifier, &DiveListNotifier::numShownChanged, this, &StatsView::replotIfVisible);
|
|
|
|
chart->installEventFilter(&eventFilter);
|
|
chart->setAcceptHoverEvents(true);
|
|
chart->legend()->setVisible(false);
|
|
}
|
|
|
|
QFont font;
|
|
titleFont = QFont(font.family(), font.pointSize(), QFont::Light); // Make configurable
|
|
}
|
|
|
|
StatsView::~StatsView()
|
|
{
|
|
}
|
|
|
|
void StatsView::plotAreaChanged(const QRectF &r)
|
|
{
|
|
double left = r.x() + sceneBorder;
|
|
double top = r.y() + sceneBorder;
|
|
double right = r.right() - sceneBorder;
|
|
double bottom = r.bottom() - sceneBorder;
|
|
const double minSize = 30.0;
|
|
|
|
if (title)
|
|
top += title->boundingRect().height() + titleBorder;
|
|
// Currently, we only have either none, or an x- and a y-axis
|
|
if (xAxis)
|
|
bottom -= xAxis->height();
|
|
if (bottom - top < minSize)
|
|
return;
|
|
if (yAxis) {
|
|
yAxis->setSize(bottom - top);
|
|
left += yAxis->width();
|
|
yAxis->setPos(QPointF(left, bottom));
|
|
}
|
|
if (right - left < minSize)
|
|
return;
|
|
if (xAxis) {
|
|
xAxis->setSize(right - left);
|
|
xAxis->setPos(QPointF(left, bottom));
|
|
}
|
|
|
|
if (grid)
|
|
grid->updatePositions();
|
|
for (auto &series: series)
|
|
series->updatePositions();
|
|
for (QuartileMarker &marker: quartileMarkers)
|
|
marker.updatePosition();
|
|
for (RegressionLine &line: regressionLines)
|
|
line.updatePosition();
|
|
for (HistogramMarker &marker: histogramMarkers)
|
|
marker.updatePosition();
|
|
if (legend)
|
|
legend->resize();
|
|
updateTitlePos();
|
|
}
|
|
|
|
void StatsView::replotIfVisible()
|
|
{
|
|
if (isVisible())
|
|
plot(state);
|
|
}
|
|
|
|
void StatsView::hover(QPointF pos)
|
|
{
|
|
for (auto &series: series) {
|
|
if (series->hover(pos)) {
|
|
if (series.get() != highlightedSeries) {
|
|
if (highlightedSeries)
|
|
highlightedSeries->unhighlight();
|
|
highlightedSeries = series.get();
|
|
}
|
|
return;
|
|
}
|
|
}
|
|
|
|
// No series was highlighted -> unhighlight any previously highlighted series.
|
|
if (highlightedSeries) {
|
|
highlightedSeries->unhighlight();
|
|
highlightedSeries = nullptr;
|
|
}
|
|
}
|
|
|
|
template <typename T, class... Args>
|
|
T *StatsView::createSeries(Args&&... args)
|
|
{
|
|
T *res = new T(chart, xAxis, yAxis, std::forward<Args>(args)...);
|
|
series.emplace_back(res);
|
|
series.back()->updatePositions();
|
|
return res;
|
|
}
|
|
|
|
void StatsView::setTitle(const QString &s)
|
|
{
|
|
if (s.isEmpty()) {
|
|
title.reset();
|
|
return;
|
|
}
|
|
title = std::make_unique<QGraphicsSimpleTextItem>(s, chart);
|
|
title->setFont(titleFont);
|
|
}
|
|
|
|
void StatsView::updateTitlePos()
|
|
{
|
|
if (!title)
|
|
return;
|
|
QRectF rect = chart->plotArea();
|
|
title->setPos(sceneBorder + (rect.width() - title->boundingRect().width()) / 2.0,
|
|
sceneBorder);
|
|
}
|
|
|
|
template <typename T, class... Args>
|
|
T *StatsView::createAxis(const QString &title, Args&&... args)
|
|
{
|
|
T *res = new T(chart, title, std::forward<Args>(args)...);
|
|
axes.emplace_back(res);
|
|
return res;
|
|
}
|
|
|
|
void StatsView::setAxes(StatsAxis *x, StatsAxis *y)
|
|
{
|
|
xAxis = x;
|
|
yAxis = y;
|
|
if (x && y)
|
|
grid = std::make_unique<StatsGrid>(chart, *x, *y);
|
|
}
|
|
|
|
void StatsView::reset()
|
|
{
|
|
if (!chart)
|
|
return;
|
|
highlightedSeries = nullptr;
|
|
xAxis = yAxis = nullptr;
|
|
legend.reset();
|
|
series.clear();
|
|
quartileMarkers.clear();
|
|
regressionLines.clear();
|
|
histogramMarkers.clear();
|
|
chart->removeAllSeries();
|
|
grid.reset();
|
|
axes.clear();
|
|
title.reset();
|
|
}
|
|
|
|
void StatsView::plot(const StatsState &stateIn)
|
|
{
|
|
state = stateIn;
|
|
plotChart();
|
|
plotAreaChanged(chart->plotArea());
|
|
}
|
|
|
|
void StatsView::plotChart()
|
|
{
|
|
if (!chart || !state.var1)
|
|
return;
|
|
reset();
|
|
|
|
const std::vector<dive *> dives = DiveFilter::instance()->visibleDives();
|
|
switch (state.type) {
|
|
case ChartType::DiscreteBar:
|
|
return plotBarChart(dives, state.subtype, state.var1, state.var1Binner, state.var2,
|
|
state.var2Binner, state.labels, state.legend);
|
|
case ChartType::DiscreteValue:
|
|
return plotValueChart(dives, state.subtype, state.var1, state.var1Binner, state.var2,
|
|
state.var2Operation, state.labels);
|
|
case ChartType::DiscreteCount:
|
|
return plotDiscreteCountChart(dives, state.subtype, state.var1, state.var1Binner, state.labels);
|
|
case ChartType::Pie:
|
|
return plotPieChart(dives, state.var1, state.var1Binner, state.labels, state.legend);
|
|
case ChartType::DiscreteBox:
|
|
return plotDiscreteBoxChart(dives, state.var1, state.var1Binner, state.var2);
|
|
case ChartType::DiscreteScatter:
|
|
return plotDiscreteScatter(dives, state.var1, state.var1Binner, state.var2, state.quartiles);
|
|
case ChartType::HistogramCount:
|
|
return plotHistogramCountChart(dives, state.subtype, state.var1, state.var1Binner,
|
|
state.labels, state.median, state.mean);
|
|
case ChartType::HistogramValue:
|
|
return plotHistogramValueChart(dives, state.subtype, state.var1, state.var1Binner, state.var2,
|
|
state.var2Operation, state.labels);
|
|
case ChartType::HistogramStacked:
|
|
return plotHistogramStackedChart(dives, state.subtype, state.var1, state.var1Binner,
|
|
state.var2, state.var2Binner, state.labels, state.legend);
|
|
case ChartType::HistogramBox:
|
|
return plotHistogramBoxChart(dives, state.var1, state.var1Binner, state.var2);
|
|
case ChartType::ScatterPlot:
|
|
return plotScatter(dives, state.var1, state.var2);
|
|
case ChartType::Invalid:
|
|
return;
|
|
default:
|
|
qWarning("Unknown chart type: %d", (int)state.type);
|
|
return;
|
|
}
|
|
}
|
|
|
|
template<typename T>
|
|
CategoryAxis *StatsView::createCategoryAxis(const QString &name, const StatsBinner &binner,
|
|
const std::vector<T> &bins, bool isHorizontal)
|
|
{
|
|
std::vector<QString> labels;
|
|
labels.reserve(bins.size());
|
|
for (const auto &[bin, dummy]: bins)
|
|
labels.push_back(binner.format(*bin));
|
|
return createAxis<CategoryAxis>(name, labels, isHorizontal);
|
|
}
|
|
|
|
CountAxis *StatsView::createCountAxis(int maxVal, bool isHorizontal)
|
|
{
|
|
return createAxis<CountAxis>(StatsTranslations::tr("No. dives"), maxVal, isHorizontal);
|
|
}
|
|
|
|
// For "two-dimensionally" binned plots (eg. stacked bar or grouped bar):
|
|
// Counts for each bin on the independent variable, including the total counts for that bin.
|
|
struct BinCounts {
|
|
StatsBinPtr bin;
|
|
std::vector<int> counts;
|
|
int total;
|
|
};
|
|
|
|
// The problem with bar plots is that for different category
|
|
// bins, we might get different value bins. So we have to keep track
|
|
// of our counts and adjust accordingly. That's a bit annoying.
|
|
// Perhaps we should determine the bins of all dives first and then
|
|
// query the counts for precisely those bins?
|
|
struct BarPlotData {
|
|
std::vector<BinCounts> hbin_counts; // For each category bin the counts for all value bins
|
|
std::vector<StatsBinPtr> vbins;
|
|
std::vector<QString> vbinNames;
|
|
int maxCount; // Highest count of any bin-combination
|
|
int maxCategoryCount; // Highest count of any category bin
|
|
// Attention: categoryBin argument will be consumed!
|
|
BarPlotData(std::vector<StatsBinDives> &categoryBins, const StatsBinner &valuebinner);
|
|
};
|
|
|
|
BarPlotData::BarPlotData(std::vector<StatsBinDives> &categoryBins, const StatsBinner &valueBinner) :
|
|
maxCount(0), maxCategoryCount(0)
|
|
{
|
|
for (auto &[bin, dives]: categoryBins) {
|
|
// This moves the bin - the original pointer is invalidated
|
|
hbin_counts.push_back({ std::move(bin), std::vector<int>(vbins.size(), 0), 0 });
|
|
for (auto &[vbin, count]: valueBinner.count_dives(dives, false)) {
|
|
// Note: we assume that the bins are sorted!
|
|
auto it = std::lower_bound(vbins.begin(), vbins.end(), vbin,
|
|
[] (const StatsBinPtr &p, const StatsBinPtr &bin)
|
|
{ return *p < *bin; });
|
|
ssize_t pos = it - vbins.begin();
|
|
if (it == vbins.end() || **it != *vbin) {
|
|
// Add a new value bin.
|
|
// Attn: this invalidates "vbin", which must not be used henceforth!
|
|
vbins.insert(it, std::move(vbin));
|
|
// Fix the old arrays
|
|
for (auto &[bin, v, total]: hbin_counts)
|
|
v.insert(v.begin() + pos, 0);
|
|
}
|
|
hbin_counts.back().counts[pos] = count;
|
|
hbin_counts.back().total += count;
|
|
if (count > maxCount)
|
|
maxCount = count;
|
|
}
|
|
maxCategoryCount = std::max(maxCategoryCount, hbin_counts.back().total);
|
|
}
|
|
|
|
vbinNames.reserve(vbins.size());
|
|
for (const auto &vbin: vbins)
|
|
vbinNames.push_back(valueBinner.formatWithUnit(*vbin));
|
|
}
|
|
|
|
// Formats "x (y%)" as either a single or two strings for horizontal and non-horizontal cases, respectively.
|
|
static std::vector<QString> makePercentageLabels(int count, int total, bool isHorizontal)
|
|
{
|
|
double percentage = count * 100.0 / total;
|
|
QString countString = QString("%L1").arg(count);
|
|
QString percentageString = QString("%L1%").arg(percentage, 0, 'f', 1);
|
|
if (isHorizontal)
|
|
return { QString("%1 (%2)").arg(countString, percentageString) };
|
|
else
|
|
return { countString, percentageString };
|
|
}
|
|
|
|
// From a list of counts, make (count, label) pairs, where the label
|
|
// formats the total number and the percentage of dives.
|
|
static std::vector<std::pair<int, std::vector<QString>>> makeCountLabels(const std::vector<int> &counts, int total,
|
|
bool labels, bool isHorizontal)
|
|
{
|
|
std::vector<std::pair<int, std::vector<QString>>> count_labels;
|
|
count_labels.reserve(counts.size());
|
|
for (int count: counts) {
|
|
std::vector<QString> label = labels ? makePercentageLabels(count, total, isHorizontal)
|
|
: std::vector<QString>();
|
|
count_labels.push_back(std::make_pair(count, label));
|
|
}
|
|
return count_labels;
|
|
}
|
|
|
|
void StatsView::plotBarChart(const std::vector<dive *> &dives,
|
|
ChartSubType subType,
|
|
const StatsVariable *categoryVariable, const StatsBinner *categoryBinner,
|
|
const StatsVariable *valueVariable, const StatsBinner *valueBinner, bool labels, bool showLegend)
|
|
{
|
|
if (!categoryBinner || !valueBinner)
|
|
return;
|
|
|
|
setTitle(valueVariable->nameWithBinnerUnit(*valueBinner));
|
|
|
|
std::vector<StatsBinDives> categoryBins = categoryBinner->bin_dives(dives, false);
|
|
|
|
bool isStacked = subType == ChartSubType::VerticalStacked || subType == ChartSubType::HorizontalStacked;
|
|
bool isHorizontal = subType == ChartSubType::HorizontalGrouped || subType == ChartSubType::HorizontalStacked;
|
|
|
|
// Construct the histogram axis now, because the pointers to the bins
|
|
// will be moved away when constructing BarPlotData below.
|
|
CategoryAxis *catAxis = createCategoryAxis(categoryVariable->nameWithBinnerUnit(*categoryBinner),
|
|
*categoryBinner, categoryBins, !isHorizontal);
|
|
|
|
BarPlotData data(categoryBins, *valueBinner);
|
|
|
|
int maxVal = isStacked ? data.maxCategoryCount : data.maxCount;
|
|
CountAxis *valAxis = createCountAxis(maxVal, isHorizontal);
|
|
|
|
if (isHorizontal)
|
|
setAxes(valAxis, catAxis);
|
|
else
|
|
setAxes(catAxis, valAxis);
|
|
|
|
// Paint legend first, because the bin-names will be moved away from.
|
|
if (showLegend)
|
|
legend = std::make_unique<Legend>(chart, data.vbinNames);
|
|
|
|
std::vector<BarSeries::MultiItem> items;
|
|
items.reserve(data.hbin_counts.size());
|
|
double pos = 0.0;
|
|
for (auto &[hbin, counts, total]: data.hbin_counts) {
|
|
items.push_back({ pos - 0.5, pos + 0.5, makeCountLabels(counts, total, labels, isHorizontal),
|
|
categoryBinner->formatWithUnit(*hbin) });
|
|
pos += 1.0;
|
|
}
|
|
|
|
createSeries<BarSeries>(isHorizontal, isStacked, categoryVariable->name(), valueVariable, std::move(data.vbinNames), items);
|
|
}
|
|
|
|
const double NaN = std::numeric_limits<double>::quiet_NaN();
|
|
|
|
// These templates are used to extract min and max y-values of various lists.
|
|
// A bit too convoluted for my tastes - can we make that simpler?
|
|
static std::pair<double, double> getMinMaxValueBase(const std::vector<StatsValue> &values)
|
|
{
|
|
// Attention: this supposes that the list is sorted!
|
|
return values.empty() ? std::make_pair(NaN, NaN) : std::make_pair(values.front().v, values.back().v);
|
|
}
|
|
static std::pair<double, double> getMinMaxValueBase(double v)
|
|
{
|
|
return { v, v };
|
|
}
|
|
static std::pair<double, double> getMinMaxValueBase(const StatsQuartiles &q)
|
|
{
|
|
return { q.min, q.max };
|
|
}
|
|
static std::pair<double, double> getMinMaxValueBase(const StatsScatterItem &s)
|
|
{
|
|
return { s.y, s.y };
|
|
}
|
|
template <typename T1, typename T2>
|
|
static std::pair<double, double> getMinMaxValueBase(const std::pair<T1, T2> &p)
|
|
{
|
|
return getMinMaxValueBase(p.second);
|
|
}
|
|
template <typename T>
|
|
static std::pair<double, double> getMinMaxValueBase(const StatsBinValue<T> &v)
|
|
{
|
|
return getMinMaxValueBase(v.value);
|
|
}
|
|
|
|
template <typename T>
|
|
static void updateMinMax(double &min, double &max, bool &found, const T &v)
|
|
{
|
|
const auto [mi, ma] = getMinMaxValueBase(v);
|
|
if (!std::isnan(mi) && mi < min)
|
|
min = mi;
|
|
if (!std::isnan(ma) && ma > max)
|
|
max = ma;
|
|
if (!std::isnan(mi) || !std::isnan(ma))
|
|
found = true;
|
|
}
|
|
|
|
template <typename T>
|
|
static std::pair<double, double> getMinMaxValue(const std::vector<T> &values)
|
|
{
|
|
double min = 1e14, max = 0.0;
|
|
bool found = false;
|
|
for (const T &v: values)
|
|
updateMinMax(min, max, found, v);
|
|
return found ? std::make_pair(min, max) : std::make_pair(0.0, 0.0);
|
|
}
|
|
|
|
static std::pair<double, double> getMinMaxValue(const std::vector<StatsBinOp> &bins, StatsOperation op)
|
|
{
|
|
double min = 1e14, max = 0.0;
|
|
bool found = false;
|
|
for (auto &[bin, res]: bins) {
|
|
if (!res.isValid())
|
|
continue;
|
|
updateMinMax(min, max, found, res.get(op));
|
|
}
|
|
return found ? std::make_pair(min, max) : std::make_pair(0.0, 0.0);
|
|
}
|
|
|
|
void StatsView::plotValueChart(const std::vector<dive *> &dives,
|
|
ChartSubType subType,
|
|
const StatsVariable *categoryVariable, const StatsBinner *categoryBinner,
|
|
const StatsVariable *valueVariable, StatsOperation valueAxisOperation,
|
|
bool labels)
|
|
{
|
|
if (!categoryBinner)
|
|
return;
|
|
|
|
setTitle(QStringLiteral("%1 (%2)").arg(valueVariable->name(), StatsVariable::operationName(valueAxisOperation)));
|
|
|
|
std::vector<StatsBinOp> categoryBins = valueVariable->bin_operations(*categoryBinner, dives, false);
|
|
|
|
// If there is nothing to display, quit
|
|
if (categoryBins.empty())
|
|
return;
|
|
|
|
|
|
bool isHorizontal = subType == ChartSubType::Horizontal;
|
|
const auto [minValue, maxValue] = getMinMaxValue(categoryBins, valueAxisOperation);
|
|
int decimals = valueVariable->decimals();
|
|
CategoryAxis *catAxis = createCategoryAxis(categoryVariable->nameWithBinnerUnit(*categoryBinner),
|
|
*categoryBinner, categoryBins, !isHorizontal);
|
|
ValueAxis *valAxis = createAxis<ValueAxis>(valueVariable->nameWithUnit(),
|
|
0.0, maxValue, valueVariable->decimals(), isHorizontal);
|
|
|
|
if (isHorizontal)
|
|
setAxes(valAxis, catAxis);
|
|
else
|
|
setAxes(catAxis, valAxis);
|
|
|
|
std::vector<BarSeries::ValueItem> items;
|
|
items.reserve(categoryBins.size());
|
|
double pos = 0.0;
|
|
QString unit = valueVariable->unitSymbol();
|
|
for (auto &[bin, res]: categoryBins) {
|
|
if (res.isValid()) {
|
|
double height = res.get(valueAxisOperation);
|
|
QString value = QString("%L1").arg(height, 0, 'f', decimals);
|
|
std::vector<QString> label = labels ? std::vector<QString> { value }
|
|
: std::vector<QString>();
|
|
items.push_back({ pos - 0.5, pos + 0.5, height, label,
|
|
categoryBinner->formatWithUnit(*bin), res });
|
|
}
|
|
pos += 1.0;
|
|
}
|
|
|
|
createSeries<BarSeries>(isHorizontal, categoryVariable->name(), valueVariable, items);
|
|
}
|
|
|
|
static int getTotalCount(const std::vector<StatsBinCount> &bins)
|
|
{
|
|
int total = 0;
|
|
for (const auto &[bin, count]: bins)
|
|
total += count;
|
|
return total;
|
|
}
|
|
|
|
template<typename T>
|
|
static int getMaxCount(const std::vector<T> &bins)
|
|
{
|
|
int res = 0;
|
|
for (auto const &[dummy, val]: bins) {
|
|
if (val > res)
|
|
res = val;
|
|
}
|
|
return res;
|
|
}
|
|
|
|
void StatsView::plotDiscreteCountChart(const std::vector<dive *> &dives,
|
|
ChartSubType subType,
|
|
const StatsVariable *categoryVariable, const StatsBinner *categoryBinner,
|
|
bool labels)
|
|
{
|
|
if (!categoryBinner)
|
|
return;
|
|
|
|
setTitle(categoryVariable->nameWithBinnerUnit(*categoryBinner));
|
|
|
|
std::vector<StatsBinCount> categoryBins = categoryBinner->count_dives(dives, false);
|
|
|
|
// If there is nothing to display, quit
|
|
if (categoryBins.empty())
|
|
return;
|
|
|
|
int total = getTotalCount(categoryBins);
|
|
bool isHorizontal = subType != ChartSubType::Vertical;
|
|
|
|
CategoryAxis *catAxis = createCategoryAxis(categoryVariable->nameWithBinnerUnit(*categoryBinner),
|
|
*categoryBinner, categoryBins, !isHorizontal);
|
|
|
|
int maxCount = getMaxCount(categoryBins);
|
|
CountAxis *valAxis = createCountAxis(maxCount, isHorizontal);
|
|
|
|
if (isHorizontal)
|
|
setAxes(valAxis, catAxis);
|
|
else
|
|
setAxes(catAxis, valAxis);
|
|
|
|
std::vector<BarSeries::CountItem> items;
|
|
items.reserve(categoryBins.size());
|
|
double pos = 0.0;
|
|
for (auto const &[bin, count]: categoryBins) {
|
|
std::vector<QString> label = labels ? makePercentageLabels(count, total, isHorizontal)
|
|
: std::vector<QString>();
|
|
items.push_back({ pos - 0.5, pos + 0.5, count, label,
|
|
categoryBinner->formatWithUnit(*bin), total });
|
|
pos += 1.0;
|
|
}
|
|
|
|
createSeries<BarSeries>(isHorizontal, categoryVariable->name(), items);
|
|
}
|
|
|
|
void StatsView::plotPieChart(const std::vector<dive *> &dives,
|
|
const StatsVariable *categoryVariable, const StatsBinner *categoryBinner,
|
|
bool labels, bool showLegend)
|
|
{
|
|
if (!categoryBinner)
|
|
return;
|
|
|
|
setTitle(categoryVariable->nameWithBinnerUnit(*categoryBinner));
|
|
|
|
std::vector<StatsBinCount> categoryBins = categoryBinner->count_dives(dives, false);
|
|
|
|
// If there is nothing to display, quit
|
|
if (categoryBins.empty())
|
|
return;
|
|
|
|
std::vector<std::pair<QString, int>> data;
|
|
data.reserve(categoryBins.size());
|
|
for (auto const &[bin, count]: categoryBins)
|
|
data.emplace_back(categoryBinner->formatWithUnit(*bin), count);
|
|
|
|
bool keepOrder = categoryVariable->type() != StatsVariable::Type::Discrete;
|
|
PieSeries *series = createSeries<PieSeries>(categoryVariable->name(), data, keepOrder, labels);
|
|
|
|
if (showLegend)
|
|
legend = std::make_unique<Legend>(chart, series->binNames());
|
|
}
|
|
|
|
void StatsView::plotDiscreteBoxChart(const std::vector<dive *> &dives,
|
|
const StatsVariable *categoryVariable, const StatsBinner *categoryBinner,
|
|
const StatsVariable *valueVariable)
|
|
{
|
|
if (!categoryBinner)
|
|
return;
|
|
|
|
setTitle(valueVariable->name());
|
|
|
|
std::vector<StatsBinQuartiles> categoryBins = valueVariable->bin_quartiles(*categoryBinner, dives, false);
|
|
|
|
// If there is nothing to display, quit
|
|
if (categoryBins.empty())
|
|
return;
|
|
|
|
CategoryAxis *catAxis = createCategoryAxis(categoryVariable->nameWithBinnerUnit(*categoryBinner),
|
|
*categoryBinner, categoryBins, true);
|
|
|
|
auto [minY, maxY] = getMinMaxValue(categoryBins);
|
|
ValueAxis *valueAxis = createAxis<ValueAxis>(valueVariable->nameWithUnit(),
|
|
minY, maxY, valueVariable->decimals(), false);
|
|
|
|
setAxes(catAxis, valueAxis);
|
|
|
|
BoxSeries *series = createSeries<BoxSeries>(valueVariable->name(), valueVariable->unitSymbol(), valueVariable->decimals());
|
|
|
|
double pos = 0.0;
|
|
for (auto &[bin, q]: categoryBins) {
|
|
if (q.isValid())
|
|
series->append(pos - 0.5, pos + 0.5, q, categoryBinner->formatWithUnit(*bin));
|
|
pos += 1.0;
|
|
}
|
|
}
|
|
|
|
void StatsView::plotDiscreteScatter(const std::vector<dive *> &dives,
|
|
const StatsVariable *categoryVariable, const StatsBinner *categoryBinner,
|
|
const StatsVariable *valueVariable, bool quartiles)
|
|
{
|
|
if (!categoryBinner)
|
|
return;
|
|
|
|
setTitle(valueVariable->name());
|
|
|
|
std::vector<StatsBinValues> categoryBins = valueVariable->bin_values(*categoryBinner, dives, false);
|
|
|
|
// If there is nothing to display, quit
|
|
if (categoryBins.empty())
|
|
return;
|
|
|
|
CategoryAxis *catAxis = createCategoryAxis(categoryVariable->nameWithBinnerUnit(*categoryBinner),
|
|
*categoryBinner, categoryBins, true);
|
|
|
|
auto [minValue, maxValue] = getMinMaxValue(categoryBins);
|
|
|
|
ValueAxis *valAxis = createAxis<ValueAxis>(valueVariable->nameWithUnit(),
|
|
minValue, maxValue, valueVariable->decimals(), false);
|
|
|
|
setAxes(catAxis, valAxis);
|
|
ScatterSeries *series = createSeries<ScatterSeries>(*categoryVariable, *valueVariable);
|
|
|
|
double x = 0.0;
|
|
for (const auto &[bin, array]: categoryBins) {
|
|
for (auto [v, d]: array)
|
|
series->append(d, x, v);
|
|
if (quartiles) {
|
|
StatsQuartiles quartiles = StatsVariable::quartiles(array);
|
|
if (quartiles.isValid()) {
|
|
quartileMarkers.emplace_back(x, quartiles.q1, chart, catAxis, valAxis);
|
|
quartileMarkers.emplace_back(x, quartiles.q2, chart, catAxis, valAxis);
|
|
quartileMarkers.emplace_back(x, quartiles.q3, chart, catAxis, valAxis);
|
|
}
|
|
}
|
|
x += 1.0;
|
|
}
|
|
}
|
|
|
|
StatsView::QuartileMarker::QuartileMarker(double pos, double value, QtCharts::QChart *chart, StatsAxis *xAxis, StatsAxis *yAxis) :
|
|
item(new QGraphicsLineItem(chart)),
|
|
xAxis(xAxis), yAxis(yAxis),
|
|
pos(pos),
|
|
value(value)
|
|
{
|
|
item->setZValue(ZValues::chartFeatures);
|
|
item->setPen(QPen(quartileMarkerColor, 2.0));
|
|
updatePosition();
|
|
}
|
|
|
|
void StatsView::QuartileMarker::updatePosition()
|
|
{
|
|
if (!xAxis || !yAxis)
|
|
return;
|
|
double x = xAxis->toScreen(pos);
|
|
double y = yAxis->toScreen(value);
|
|
item->setLine(x - quartileMarkerSize / 2.0, y,
|
|
x + quartileMarkerSize / 2.0, y);
|
|
}
|
|
|
|
StatsView::RegressionLine::RegressionLine(double a, double b, QPen pen, QtCharts::QChart *chart, StatsAxis *xAxis, StatsAxis *yAxis) :
|
|
item(new QGraphicsLineItem(chart)),
|
|
xAxis(xAxis), yAxis(yAxis),
|
|
a(a), b(b)
|
|
{
|
|
item->setZValue(ZValues::chartFeatures);
|
|
item->setPen(pen);
|
|
}
|
|
|
|
void StatsView::RegressionLine::updatePosition()
|
|
{
|
|
if (!xAxis || !yAxis)
|
|
return;
|
|
auto [minX, maxX] = xAxis->minMax();
|
|
auto [minY, maxY] = yAxis->minMax();
|
|
double y1 = a * minX + b;
|
|
double y2 = a * maxX + b;
|
|
|
|
// If not fully inside drawing region, do clipping.
|
|
if ((y1 < minY || y1 > maxY || y2 < minY || y2 > maxY) && fabs(a) > 0.0001) {
|
|
// Intersections with y = minY and y = maxY lines
|
|
double intersect_x1 = (minY - b) / a;
|
|
double intersect_x2 = (maxY - b) / a;
|
|
if (intersect_x1 > intersect_x2)
|
|
std::swap(intersect_x1, intersect_x2);
|
|
minX = std::max(minX, intersect_x1);
|
|
maxX = std::min(maxX, intersect_x2);
|
|
}
|
|
item->setLine(xAxis->toScreen(minX), yAxis->toScreen(a * minX + b),
|
|
xAxis->toScreen(maxX), yAxis->toScreen(a * maxX + b));
|
|
}
|
|
|
|
StatsView::HistogramMarker::HistogramMarker(double val, bool horizontal, QPen pen, QtCharts::QChart *chart, StatsAxis *xAxis, StatsAxis *yAxis) :
|
|
item(new QGraphicsLineItem(chart)),
|
|
xAxis(xAxis), yAxis(yAxis),
|
|
val(val), horizontal(horizontal)
|
|
{
|
|
item->setZValue(ZValues::chartFeatures);
|
|
item->setPen(pen);
|
|
}
|
|
|
|
void StatsView::HistogramMarker::updatePosition()
|
|
{
|
|
if (!xAxis || !yAxis)
|
|
return;
|
|
if (horizontal) {
|
|
double y = yAxis->toScreen(val);
|
|
auto [x1, x2] = xAxis->minMaxScreen();
|
|
item->setLine(x1, y, x2, y);
|
|
} else {
|
|
double x = xAxis->toScreen(val);
|
|
auto [y1, y2] = yAxis->minMaxScreen();
|
|
item->setLine(x, y1, x, y2);
|
|
}
|
|
}
|
|
|
|
void StatsView::addHistogramMarker(double pos, const QPen &pen, bool isHorizontal, StatsAxis *xAxis, StatsAxis *yAxis)
|
|
{
|
|
histogramMarkers.emplace_back(pos, isHorizontal, pen, chart, xAxis, yAxis);
|
|
}
|
|
|
|
void StatsView::addLinearRegression(double a, double b, double minX, double maxX, double minY, double maxY, StatsAxis *xAxis, StatsAxis *yAxis)
|
|
{
|
|
regressionLines.emplace_back(a, b, QPen(Qt::red), chart, xAxis, yAxis);
|
|
}
|
|
|
|
// Yikes, we get our data in different kinds of (bin, value) pairs.
|
|
// To create a category axis from this, we have to templatify the function.
|
|
template<typename T>
|
|
HistogramAxis *StatsView::createHistogramAxis(const QString &name, const StatsBinner &binner,
|
|
const std::vector<T> &bins, bool isHorizontal)
|
|
{
|
|
std::vector<HistogramAxisEntry> labels;
|
|
for (auto const &[bin, dummy]: bins) {
|
|
QString label = binner.formatLowerBound(*bin);
|
|
double lowerBound = binner.lowerBoundToFloat(*bin);
|
|
bool prefer = binner.preferBin(*bin);
|
|
labels.push_back({ label, lowerBound, prefer });
|
|
}
|
|
|
|
const StatsBin &lastBin = *bins.back().bin;
|
|
QString lastLabel = binner.formatUpperBound(lastBin);
|
|
double upperBound = binner.upperBoundToFloat(lastBin);
|
|
labels.push_back({ lastLabel, upperBound, false });
|
|
|
|
return createAxis<HistogramAxis>(name, std::move(labels), isHorizontal);
|
|
}
|
|
|
|
void StatsView::plotHistogramCountChart(const std::vector<dive *> &dives,
|
|
ChartSubType subType,
|
|
const StatsVariable *categoryVariable, const StatsBinner *categoryBinner,
|
|
bool labels, bool showMedian, bool showMean)
|
|
{
|
|
if (!categoryBinner)
|
|
return;
|
|
|
|
setTitle(categoryVariable->name());
|
|
|
|
std::vector<StatsBinCount> categoryBins = categoryBinner->count_dives(dives, true);
|
|
|
|
// If there is nothing to display, quit
|
|
if (categoryBins.empty())
|
|
return;
|
|
|
|
bool isHorizontal = subType == ChartSubType::Horizontal;
|
|
HistogramAxis *catAxis = createHistogramAxis(categoryVariable->nameWithBinnerUnit(*categoryBinner),
|
|
*categoryBinner, categoryBins, !isHorizontal);
|
|
|
|
int maxCategoryCount = getMaxCount(categoryBins);
|
|
int total = getTotalCount(categoryBins);
|
|
|
|
StatsAxis *valAxis = createCountAxis(maxCategoryCount, isHorizontal);
|
|
|
|
if (isHorizontal)
|
|
setAxes(valAxis, catAxis);
|
|
else
|
|
setAxes(catAxis, valAxis);
|
|
|
|
std::vector<BarSeries::CountItem> items;
|
|
items.reserve(categoryBins.size());
|
|
|
|
for (auto const &[bin, count]: categoryBins) {
|
|
double lowerBound = categoryBinner->lowerBoundToFloat(*bin);
|
|
double upperBound = categoryBinner->upperBoundToFloat(*bin);
|
|
std::vector<QString> label = labels ? makePercentageLabels(count, total, isHorizontal)
|
|
: std::vector<QString>();
|
|
|
|
items.push_back({ lowerBound, upperBound, count, label,
|
|
categoryBinner->formatWithUnit(*bin), total });
|
|
}
|
|
|
|
createSeries<BarSeries>(isHorizontal, categoryVariable->name(), items);
|
|
|
|
if (categoryVariable->type() == StatsVariable::Type::Numeric) {
|
|
if (showMean) {
|
|
double mean = categoryVariable->mean(dives);
|
|
QPen meanPen(Qt::green);
|
|
meanPen.setWidth(2);
|
|
if (!std::isnan(mean))
|
|
addHistogramMarker(mean, meanPen, isHorizontal, xAxis, yAxis);
|
|
}
|
|
if (showMedian) {
|
|
double median = categoryVariable->quartiles(dives).q2;
|
|
QPen medianPen(Qt::red);
|
|
medianPen.setWidth(2);
|
|
if (!std::isnan(median))
|
|
addHistogramMarker(median, medianPen, isHorizontal, xAxis, yAxis);
|
|
}
|
|
}
|
|
}
|
|
|
|
void StatsView::plotHistogramValueChart(const std::vector<dive *> &dives,
|
|
ChartSubType subType,
|
|
const StatsVariable *categoryVariable, const StatsBinner *categoryBinner,
|
|
const StatsVariable *valueVariable, StatsOperation valueAxisOperation,
|
|
bool labels)
|
|
{
|
|
if (!categoryBinner)
|
|
return;
|
|
|
|
setTitle(QStringLiteral("%1 (%2)").arg(valueVariable->name(), StatsVariable::operationName(valueAxisOperation)));
|
|
|
|
std::vector<StatsBinOp> categoryBins = valueVariable->bin_operations(*categoryBinner, dives, true);
|
|
|
|
// If there is nothing to display, quit
|
|
if (categoryBins.empty())
|
|
return;
|
|
|
|
bool isHorizontal = subType == ChartSubType::Horizontal;
|
|
HistogramAxis *catAxis = createHistogramAxis(categoryVariable->nameWithBinnerUnit(*categoryBinner),
|
|
*categoryBinner, categoryBins, !isHorizontal);
|
|
|
|
const auto [minValue, maxValue] = getMinMaxValue(categoryBins, valueAxisOperation);
|
|
|
|
int decimals = valueVariable->decimals();
|
|
ValueAxis *valAxis = createAxis<ValueAxis>(valueVariable->nameWithUnit(),
|
|
0.0, maxValue, decimals, isHorizontal);
|
|
|
|
if (isHorizontal)
|
|
setAxes(valAxis, catAxis);
|
|
else
|
|
setAxes(catAxis, valAxis);
|
|
|
|
std::vector<BarSeries::ValueItem> items;
|
|
items.reserve(categoryBins.size());
|
|
|
|
QString unit = valueVariable->unitSymbol();
|
|
for (auto const &[bin, res]: categoryBins) {
|
|
if (!res.isValid())
|
|
continue;
|
|
double height = res.get(valueAxisOperation);
|
|
double lowerBound = categoryBinner->lowerBoundToFloat(*bin);
|
|
double upperBound = categoryBinner->upperBoundToFloat(*bin);
|
|
QString value = QString("%L1").arg(height, 0, 'f', decimals);
|
|
std::vector<QString> label = labels ? std::vector<QString> { value }
|
|
: std::vector<QString>();
|
|
items.push_back({ lowerBound, upperBound, height, label,
|
|
categoryBinner->formatWithUnit(*bin), res });
|
|
}
|
|
|
|
createSeries<BarSeries>(isHorizontal, categoryVariable->name(), valueVariable, items);
|
|
}
|
|
|
|
void StatsView::plotHistogramStackedChart(const std::vector<dive *> &dives,
|
|
ChartSubType subType,
|
|
const StatsVariable *categoryVariable, const StatsBinner *categoryBinner,
|
|
const StatsVariable *valueVariable, const StatsBinner *valueBinner, bool labels, bool showLegend)
|
|
{
|
|
if (!categoryBinner || !valueBinner)
|
|
return;
|
|
|
|
setTitle(valueVariable->nameWithBinnerUnit(*valueBinner));
|
|
|
|
std::vector<StatsBinDives> categoryBins = categoryBinner->bin_dives(dives, true);
|
|
|
|
// Construct the histogram axis now, because the pointers to the bins
|
|
// will be moved away when constructing BarPlotData below.
|
|
bool isHorizontal = subType == ChartSubType::HorizontalStacked;
|
|
HistogramAxis *catAxis = createHistogramAxis(categoryVariable->nameWithBinnerUnit(*categoryBinner),
|
|
*categoryBinner, categoryBins, !isHorizontal);
|
|
|
|
BarPlotData data(categoryBins, *valueBinner);
|
|
if (showLegend)
|
|
legend = std::make_unique<Legend>(chart, data.vbinNames);
|
|
|
|
CountAxis *valAxis = createCountAxis(data.maxCategoryCount, isHorizontal);
|
|
|
|
if (isHorizontal)
|
|
setAxes(valAxis, catAxis);
|
|
else
|
|
setAxes(catAxis, valAxis);
|
|
|
|
std::vector<BarSeries::MultiItem> items;
|
|
items.reserve(data.hbin_counts.size());
|
|
|
|
for (auto &[hbin, counts, total]: data.hbin_counts) {
|
|
double lowerBound = categoryBinner->lowerBoundToFloat(*hbin);
|
|
double upperBound = categoryBinner->upperBoundToFloat(*hbin);
|
|
items.push_back({ lowerBound, upperBound, makeCountLabels(counts, total, labels, isHorizontal),
|
|
categoryBinner->formatWithUnit(*hbin) });
|
|
}
|
|
|
|
createSeries<BarSeries>(isHorizontal, true, categoryVariable->name(), valueVariable, std::move(data.vbinNames), items);
|
|
}
|
|
|
|
void StatsView::plotHistogramBoxChart(const std::vector<dive *> &dives,
|
|
const StatsVariable *categoryVariable, const StatsBinner *categoryBinner,
|
|
const StatsVariable *valueVariable)
|
|
{
|
|
if (!categoryBinner)
|
|
return;
|
|
|
|
setTitle(valueVariable->name());
|
|
|
|
std::vector<StatsBinQuartiles> categoryBins = valueVariable->bin_quartiles(*categoryBinner, dives, true);
|
|
|
|
// If there is nothing to display, quit
|
|
if (categoryBins.empty())
|
|
return;
|
|
|
|
HistogramAxis *catAxis = createHistogramAxis(categoryVariable->nameWithBinnerUnit(*categoryBinner),
|
|
*categoryBinner, categoryBins, true);
|
|
|
|
auto [minY, maxY] = getMinMaxValue(categoryBins);
|
|
ValueAxis *valueAxis = createAxis<ValueAxis>(valueVariable->nameWithUnit(),
|
|
minY, maxY, valueVariable->decimals(), false);
|
|
|
|
setAxes(catAxis, valueAxis);
|
|
|
|
BoxSeries *series = createSeries<BoxSeries>(valueVariable->name(), valueVariable->unitSymbol(), valueVariable->decimals());
|
|
|
|
for (auto &[bin, q]: categoryBins) {
|
|
if (!q.isValid())
|
|
continue;
|
|
double lowerBound = categoryBinner->lowerBoundToFloat(*bin);
|
|
double upperBound = categoryBinner->upperBoundToFloat(*bin);
|
|
series->append(lowerBound, upperBound, q, categoryBinner->formatWithUnit(*bin));
|
|
}
|
|
}
|
|
|
|
static bool is_linear_regression(int sample_size, double cov, double sx2, double sy2)
|
|
{
|
|
// One point never, two points always form a line
|
|
if (sample_size < 2)
|
|
return false;
|
|
if (sample_size <= 2)
|
|
return true;
|
|
|
|
const double tval[] = { 12.709, 4.303, 3.182, 2.776, 2.571, 2.447, 2.201, 2.120, 2.080, 2.056, 2.021, 1.960, 1.960 };
|
|
const int t_df[] = { 1, 2, 3, 4, 5, 6, 11, 16, 21, 26, 40, 100, 100000 };
|
|
int df = sample_size - 2; // Following is the one-tailed t-value at p < 0.05 and [sample_size - 2] degrees of freedom for the dive data:
|
|
double t = (cov / sx2) / sqrt(((sy2 - cov * cov / sx2) / (double)df) / sx2);
|
|
for (int i = std::size(tval) - 2; i >= 0; i--) { // We do linear interpolation rather than having a large lookup table.
|
|
if (df >= t_df[i]) { // Look up the appropriate reference t-value at p < 0.05 and df degrees of freedom
|
|
double t_lookup = tval[i] - (tval[i] - tval[i+1]) * (df - t_df[i]) / (t_df[i+1] - t_df[i]);
|
|
return abs(t) >= t_lookup;
|
|
}
|
|
}
|
|
|
|
return true; // can't happen, as we tested for sample_size above.
|
|
}
|
|
|
|
// Returns the coefficients [a,b] of the line y = ax + b
|
|
// If case of an undetermined regression or one with infinite slope, returns [nan, nan]
|
|
static std::pair<double, double> linear_regression(const std::vector<StatsScatterItem> &v)
|
|
{
|
|
if (v.size() < 2)
|
|
return { NaN, NaN };
|
|
|
|
// First, calculate the x and y average
|
|
double avg_x = 0.0, avg_y = 0.0;
|
|
for (auto [x, y, d]: v) {
|
|
avg_x += x;
|
|
avg_y += y;
|
|
}
|
|
avg_x /= (double)v.size();
|
|
avg_y /= (double)v.size();
|
|
|
|
double cov = 0.0, sx2 = 0.0, sy2 = 0.0;
|
|
for (auto [x, y, d]: v) {
|
|
cov += (x - avg_x) * (y - avg_y);
|
|
sx2 += (x - avg_x) * (x - avg_x);
|
|
sy2 += (y - avg_y) * (y - avg_y);
|
|
}
|
|
|
|
bool is_linear = is_linear_regression((int)v.size(), cov, sx2, sy2);
|
|
|
|
if (fabs(sx2) < 1e-10 || !is_linear) // If t is not statistically significant, do not plot the regression line.
|
|
return { NaN, NaN };
|
|
double a = cov / sx2;
|
|
double b = avg_y - a * avg_x;
|
|
return { a, b };
|
|
}
|
|
|
|
void StatsView::plotScatter(const std::vector<dive *> &dives, const StatsVariable *categoryVariable, const StatsVariable *valueVariable)
|
|
{
|
|
setTitle(StatsTranslations::tr("%1 vs. %2").arg(valueVariable->name(), categoryVariable->name()));
|
|
|
|
std::vector<StatsScatterItem> points = categoryVariable->scatter(*valueVariable, dives);
|
|
if (points.empty())
|
|
return;
|
|
|
|
double minX = points.front().x;
|
|
double maxX = points.back().x;
|
|
auto [minY, maxY] = getMinMaxValue(points);
|
|
|
|
StatsAxis *axisX = categoryVariable->type() == StatsVariable::Type::Continuous ?
|
|
static_cast<StatsAxis *>(createAxis<DateAxis>(categoryVariable->nameWithUnit(),
|
|
minX, maxX, true)) :
|
|
static_cast<StatsAxis *>(createAxis<ValueAxis>(categoryVariable->nameWithUnit(),
|
|
minX, maxX, categoryVariable->decimals(), true));
|
|
|
|
StatsAxis *axisY = createAxis<ValueAxis>(valueVariable->nameWithUnit(), minY, maxY, valueVariable->decimals(), false);
|
|
|
|
setAxes(axisX, axisY);
|
|
ScatterSeries *series = createSeries<ScatterSeries>(*categoryVariable, *valueVariable);
|
|
|
|
for (auto [x, y, dive]: points)
|
|
series->append(dive, x, y);
|
|
|
|
// y = ax + b
|
|
auto [a, b] = linear_regression(points);
|
|
if (!std::isnan(a)) {
|
|
auto [minx, maxx] = axisX->minMax();
|
|
auto [miny, maxy] = axisY->minMax();
|
|
addLinearRegression(a, b, minx, maxx, miny, maxy, xAxis, yAxis);
|
|
}
|
|
}
|