mirror of
https://github.com/subsurface/subsurface.git
synced 2024-11-30 22:20:21 +00:00
statistics: implement primitive "restrict to selection" feature
Allow the user to restrict the analyzed dives based on the current selection. One button restricts to the current selection and one button resets the restriction. Thus, the user can for example select bars in the bar chart or a range in the scatter plot and perform statistics on these sets. The restriction works on top of the filter. The UI can certainly be improved, but it is a start. Signed-off-by: Berthold Stoeger <bstoeger@mail.tuwien.ac.at>
This commit is contained in:
parent
2943b1cbde
commit
5a8d7617ce
5 changed files with 97 additions and 3 deletions
|
@ -84,6 +84,8 @@ StatsWidget::StatsWidget(QWidget *parent) : QWidget(parent)
|
|||
connect(ui.var1Binner, QOverload<int>::of(&QComboBox::activated), this, &StatsWidget::var1BinnerChanged);
|
||||
connect(ui.var2Binner, QOverload<int>::of(&QComboBox::activated), this, &StatsWidget::var2BinnerChanged);
|
||||
connect(ui.var2Operation, QOverload<int>::of(&QComboBox::activated), this, &StatsWidget::var2OperationChanged);
|
||||
connect(ui.restrictButton, &QToolButton::clicked, this, &StatsWidget::restrict);
|
||||
connect(ui.unrestrictButton, &QToolButton::clicked, this, &StatsWidget::unrestrict);
|
||||
|
||||
ui.stats->setSource(urlStatsView);
|
||||
ui.stats->setResizeMode(QQuickWidget::SizeRootObjectToView);
|
||||
|
@ -141,6 +143,18 @@ void StatsWidget::updateUi()
|
|||
view->plot(state);
|
||||
}
|
||||
|
||||
void StatsWidget::updateRestrictionLabel()
|
||||
{
|
||||
if (!view)
|
||||
return;
|
||||
int num = view->restrictionCount();
|
||||
if (num < 0)
|
||||
ui.restrictionLabel->setText(tr("Analyzing all dives"));
|
||||
else
|
||||
ui.restrictionLabel->setText(tr("Analyzing subset (%L1) dives").arg(num));
|
||||
ui.unrestrictButton->setEnabled(num > 0);
|
||||
}
|
||||
|
||||
void StatsWidget::closeStats()
|
||||
{
|
||||
MainWindow::instance()->setApplicationState(ApplicationState::Default);
|
||||
|
@ -192,6 +206,21 @@ void StatsWidget::featureChanged(int idx, bool status)
|
|||
|
||||
void StatsWidget::showEvent(QShowEvent *e)
|
||||
{
|
||||
unrestrict();
|
||||
updateUi();
|
||||
QWidget::showEvent(e);
|
||||
}
|
||||
|
||||
void StatsWidget::restrict()
|
||||
{
|
||||
if (view)
|
||||
view->restrictToSelection();
|
||||
updateRestrictionLabel();
|
||||
}
|
||||
|
||||
void StatsWidget::unrestrict()
|
||||
{
|
||||
if (view)
|
||||
view->unrestrict();
|
||||
updateRestrictionLabel();
|
||||
}
|
||||
|
|
|
@ -25,11 +25,14 @@ slots:
|
|||
void var2BinnerChanged(int);
|
||||
void var2OperationChanged(int);
|
||||
void featureChanged(int, bool);
|
||||
void restrict();
|
||||
void unrestrict();
|
||||
private:
|
||||
Ui::StatsWidget ui;
|
||||
StatsState state;
|
||||
StatsView *view;
|
||||
void updateUi();
|
||||
void updateRestrictionLabel();
|
||||
std::vector<std::unique_ptr<QCheckBox>> features;
|
||||
|
||||
ChartListModel charts;
|
||||
|
|
|
@ -95,6 +95,32 @@
|
|||
</layout>
|
||||
</widget>
|
||||
</item>
|
||||
<item>
|
||||
<widget class="QGroupBox" name="restrictionGroup">
|
||||
<property name="title">
|
||||
<string>Restriction</string>
|
||||
</property>
|
||||
<layout class="QVBoxLayout" name="chartLayout">
|
||||
<item>
|
||||
<widget class="QLabel" name="restrictionLabel" />
|
||||
</item>
|
||||
<item>
|
||||
<widget class="QToolButton" name="restrictButton">
|
||||
<property name="text">
|
||||
<string>Restrict to selection</string>
|
||||
</property>
|
||||
</widget>
|
||||
</item>
|
||||
<item>
|
||||
<widget class="QToolButton" name="unrestrictButton">
|
||||
<property name="text">
|
||||
<string>Reset restriction</string>
|
||||
</property>
|
||||
</widget>
|
||||
</item>
|
||||
</layout>
|
||||
</widget>
|
||||
</item>
|
||||
<item>
|
||||
<spacer name="verticalSpacer">
|
||||
<property name="orientation">
|
||||
|
|
|
@ -39,6 +39,7 @@ StatsView::StatsView(QQuickItem *parent) : QQuickItem(parent),
|
|||
xAxis(nullptr),
|
||||
yAxis(nullptr),
|
||||
draggedItem(nullptr),
|
||||
restrictDives(false),
|
||||
rootNode(nullptr)
|
||||
{
|
||||
setFlag(ItemHasContents, true);
|
||||
|
@ -496,6 +497,25 @@ void StatsView::reset()
|
|||
grid.reset();
|
||||
}
|
||||
|
||||
void StatsView::restrictToSelection()
|
||||
{
|
||||
restrictedDives = getDiveSelection();
|
||||
std::sort(restrictedDives.begin(), restrictedDives.end()); // Sort by pointer for quick lookup
|
||||
restrictDives = true;
|
||||
plot(state);
|
||||
}
|
||||
|
||||
void StatsView::unrestrict()
|
||||
{
|
||||
restrictDives = false;
|
||||
plot(state);
|
||||
}
|
||||
|
||||
int StatsView::restrictionCount() const
|
||||
{
|
||||
return restrictDives ? (int)restrictedDives.size() : -1;
|
||||
}
|
||||
|
||||
void StatsView::plot(const StatsState &stateIn)
|
||||
{
|
||||
state = stateIn;
|
||||
|
@ -518,7 +538,19 @@ void StatsView::plotChart()
|
|||
return;
|
||||
reset();
|
||||
|
||||
const std::vector<dive *> dives = DiveFilter::instance()->visibleDives();
|
||||
std::vector<dive *> dives;
|
||||
if (restrictDives) {
|
||||
std::vector<dive *> visible = DiveFilter::instance()->visibleDives();
|
||||
dives.reserve(visible.size());
|
||||
for (dive *d: visible) {
|
||||
// binary search
|
||||
auto it = std::lower_bound(restrictedDives.begin(), restrictedDives.end(), d);
|
||||
if (it != restrictedDives.end() && *it == d)
|
||||
dives.push_back(d);
|
||||
}
|
||||
} else {
|
||||
dives = DiveFilter::instance()->visibleDives();
|
||||
}
|
||||
switch (state.type) {
|
||||
case ChartType::DiscreteBar:
|
||||
return plotBarChart(dives, state.subtype, state.var1, state.var1Binner, state.var2,
|
||||
|
|
|
@ -46,13 +46,16 @@ public:
|
|||
|
||||
void plot(const StatsState &state);
|
||||
void updateFeatures(const StatsState &state); // Updates the visibility of chart features, such as legend, regression, etc.
|
||||
void restrictToSelection();
|
||||
void unrestrict();
|
||||
int restrictionCount() const; // <0: no restriction
|
||||
QQuickWindow *w() const; // Make window available to items
|
||||
QSizeF size() const;
|
||||
QRectF plotArea() const;
|
||||
void addQSGNode(QSGNode *node, ChartZValue z); // Must only be called in render thread!
|
||||
void registerChartItem(ChartItem &item);
|
||||
void registerDirtyChartItem(ChartItem &item);
|
||||
void emergencyShutdown(); // Called when QQuick decides to delete out root node.
|
||||
void emergencyShutdown(); // Called when QQuick decides to delete our root node.
|
||||
|
||||
// Create a chart item and add it to the scene.
|
||||
// The item must not be deleted by the caller, but can be
|
||||
|
@ -131,7 +134,6 @@ private:
|
|||
// Helper functions to add feature to the chart
|
||||
void addLineMarker(double pos, double low, double high, const QPen &pen, bool isHorizontal);
|
||||
|
||||
|
||||
StatsState state;
|
||||
QFont titleFont;
|
||||
std::vector<std::unique_ptr<StatsSeries>> series;
|
||||
|
@ -148,6 +150,8 @@ private:
|
|||
QPointF dragStartMouse, dragStartItem;
|
||||
SelectionModifier selectionModifier;
|
||||
std::vector<dive *> oldSelection;
|
||||
bool restrictDives;
|
||||
std::vector<dive *> restrictedDives; // sorted by pointer for quick lookup.
|
||||
|
||||
void hoverEnterEvent(QHoverEvent *event) override;
|
||||
void hoverMoveEvent(QHoverEvent *event) override;
|
||||
|
|
Loading…
Reference in a new issue