mirror of
				https://github.com/subsurface/subsurface.git
				synced 2025-02-19 22:16:15 +00:00 
			
		
		
		
	Linear algebra class was a while ago, but somehow this does look more logical to me. Signed-off-by: Dirk Hohndel <dirk@hohndel.org>
		
			
				
	
	
		
			1008 lines
		
	
	
	
		
			34 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			1008 lines
		
	
	
	
		
			34 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 "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 <QLocale>
 | |
| 
 | |
| // Constants that control the graph layouts
 | |
| static const QColor quartileMarkerColor(Qt::red);
 | |
| static const double quartileMarkerSize = 15;
 | |
| 
 | |
| 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),
 | |
| 	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);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| StatsView::~StatsView()
 | |
| {
 | |
| }
 | |
| 
 | |
| void StatsView::plotAreaChanged(const QRectF &)
 | |
| {
 | |
| 	for (auto &axis: axes)
 | |
| 		axis->updateLabels(chart);
 | |
| 	for (auto &series: series)
 | |
| 		series->updatePositions();
 | |
| 	for (QuartileMarker &marker: quartileMarkers)
 | |
| 		marker.updatePosition();
 | |
| 	for (LineMarker &marker: lineMarkers)
 | |
| 		marker.updatePosition();
 | |
| 	if (legend)
 | |
| 		legend->resize();
 | |
| }
 | |
| 
 | |
| 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)
 | |
| {
 | |
| 	StatsAxis *xAxis = axes.size() >= 2 ? axes[0].get() : nullptr;
 | |
| 	StatsAxis *yAxis = axes.size() >= 2 ? axes[1].get() : nullptr;
 | |
| 	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)
 | |
| {
 | |
| 	chart->setTitle(s);
 | |
| }
 | |
| 
 | |
| template <typename T, class... Args>
 | |
| T *StatsView::createAxis(const QString &title, Args&&... args)
 | |
| {
 | |
| 	T *res = new T(std::forward<Args>(args)...);
 | |
| 	axes.emplace_back(res);
 | |
| 	axes.back()->updateLabels(chart);
 | |
| 	axes.back()->qaxis()->setTitleText(title);
 | |
| 	return res;
 | |
| }
 | |
| 
 | |
| void StatsView::addAxes(StatsAxis *x, StatsAxis *y)
 | |
| {
 | |
| 	chart->addAxis(x->qaxis(), Qt::AlignBottom);
 | |
| 	chart->addAxis(y->qaxis(), Qt::AlignLeft);
 | |
| }
 | |
| 
 | |
| void StatsView::reset()
 | |
| {
 | |
| 	if (!chart)
 | |
| 		return;
 | |
| 	highlightedSeries = nullptr;
 | |
| 	legend.reset();
 | |
| 	series.clear();
 | |
| 	quartileMarkers.clear();
 | |
| 	lineMarkers.clear();
 | |
| 	chart->removeAllSeries();
 | |
| 	axes.clear();
 | |
| }
 | |
| 
 | |
| void StatsView::plot(const StatsState &stateIn)
 | |
| {
 | |
| 	state = stateIn;
 | |
| 	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)
 | |
| 		addAxes(valAxis, catAxis);
 | |
| 	else
 | |
| 		addAxes(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)
 | |
| 		addAxes(valAxis, catAxis);
 | |
| 	else
 | |
| 		addAxes(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)
 | |
| 		addAxes(valAxis, catAxis);
 | |
| 	else
 | |
| 		addAxes(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);
 | |
| 
 | |
| 	addAxes(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);
 | |
| 
 | |
| 	addAxes(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, series);
 | |
| 				quartileMarkers.emplace_back(x, quartiles.q2, series);
 | |
| 				quartileMarkers.emplace_back(x, quartiles.q3, series);
 | |
| 			}
 | |
| 		}
 | |
| 		x += 1.0;
 | |
| 	}
 | |
| }
 | |
| 
 | |
| StatsView::QuartileMarker::QuartileMarker(double pos, double value, QtCharts::QAbstractSeries *series) :
 | |
| 	item(new QGraphicsLineItem(series->chart())),
 | |
| 	series(series),
 | |
| 	pos(pos),
 | |
| 	value(value)
 | |
| {
 | |
| 	item->setZValue(ZValues::chartFeatures);
 | |
| 	item->setPen(QPen(quartileMarkerColor, 2.0));
 | |
| 	updatePosition();
 | |
| }
 | |
| 
 | |
| void StatsView::QuartileMarker::updatePosition()
 | |
| {
 | |
| 	QtCharts::QChart *chart = series->chart();
 | |
| 	QPointF center = chart->mapToPosition(QPointF(pos, value), series);
 | |
| 	item->setLine(center.x() - quartileMarkerSize / 2.0, center.y(),
 | |
| 		      center.x() + quartileMarkerSize / 2.0, center.y());
 | |
| }
 | |
| 
 | |
| StatsView::LineMarker::LineMarker(QPointF from, QPointF to, QPen pen, QtCharts::QAbstractSeries *series) :
 | |
| 	item(new QGraphicsLineItem(series->chart())),
 | |
| 	series(series), from(from), to(to)
 | |
| {
 | |
| 	item->setZValue(ZValues::chartFeatures);
 | |
| 	item->setPen(pen);
 | |
| 	updatePosition();
 | |
| }
 | |
| 
 | |
| void StatsView::LineMarker::updatePosition()
 | |
| {
 | |
| 	QtCharts::QChart *chart = series->chart();
 | |
| 	item->setLine(QLineF(chart->mapToPosition(from, series),
 | |
| 			     chart->mapToPosition(to, series)));
 | |
| }
 | |
| 
 | |
| void StatsView::addLinearRegression(double a, double b, double minX, double maxX, double minY, double maxY, QtCharts::QAbstractSeries *series)
 | |
| {
 | |
| 	// Sanity check: line above or below chart
 | |
| 	double y1 = a * minX + b;
 | |
| 	double y2 = a * maxX + b;
 | |
| 	if ((y1 <= minY && y2 <= minY) || (y1 >= maxY && y2 >= maxY))
 | |
| 		return;
 | |
| 
 | |
| 	// If not fully inside drawing region, do clipping. With the check above this guarantees that a != 0,
 | |
| 	// but owing to floating point imprecision, let's test again.
 | |
| 	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);
 | |
| 	}
 | |
| 	lineMarkers.emplace_back(QPointF(minX, a * minX + b), QPointF(maxX, a * maxX + b), QPen(Qt::red), series);
 | |
| }
 | |
| 
 | |
| void StatsView::addHistogramMarker(double pos, double low, double high, const QPen &pen, bool isHorizontal, QtCharts::QAbstractSeries *series)
 | |
| {
 | |
| 	QPointF from = isHorizontal ? QPointF(low, pos) : QPointF(pos, low);
 | |
| 	QPointF to = isHorizontal ? QPointF(high, pos) : QPointF(pos, high);
 | |
| 	lineMarkers.emplace_back(from, to, pen, series);
 | |
| }
 | |
| 
 | |
| // 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);
 | |
| 	double chartHeight = valAxis->minMax().second;
 | |
| 
 | |
| 	if (isHorizontal)
 | |
| 		addAxes(valAxis, catAxis);
 | |
| 	else
 | |
| 		addAxes(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 });
 | |
| 	}
 | |
| 
 | |
| 	BarSeries *series = 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, 0.0, chartHeight, meanPen, isHorizontal, series);
 | |
| 		}
 | |
| 		if (showMedian) {
 | |
| 			double median = categoryVariable->quartiles(dives).q2;
 | |
| 			QPen medianPen(Qt::red);
 | |
| 			medianPen.setWidth(2);
 | |
| 			if (!std::isnan(median))
 | |
| 				addHistogramMarker(median, 0.0, chartHeight, medianPen, isHorizontal, series);
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| 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)
 | |
| 		addAxes(valAxis, catAxis);
 | |
| 	else
 | |
| 		addAxes(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)
 | |
| 		addAxes(valAxis, catAxis);
 | |
| 	else
 | |
| 		addAxes(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);
 | |
| 
 | |
| 	addAxes(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);
 | |
| 
 | |
| 	addAxes(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, series);
 | |
| 	}
 | |
| }
 |