summaryrefslogtreecommitdiffstats
path: root/stats/statsview.cpp
diff options
context:
space:
mode:
authorGravatar Berthold Stoeger <bstoeger@mail.tuwien.ac.at>2021-01-15 12:22:32 +0100
committerGravatar bstoeger <32835590+bstoeger@users.noreply.github.com>2021-01-20 08:47:18 +0100
commitfaf3e7079ddd680ab0e53a27a7ddc0d863792117 (patch)
tree7656180d66e75d334bef8bb2c6dd5b411631f8eb /stats/statsview.cpp
parentada5e8a49d7f0944dc3ff5ee49960298c3af9535 (diff)
downloadsubsurface-faf3e7079ddd680ab0e53a27a7ddc0d863792117.tar.gz
statistics: keep track of dirty items in double-linked list
So far the items to be recalculated in the drawing thread had a "dirty" flag and were kept in one array par z-level. Once the series are implemented in terms of QSGNodes, there may lots of these items. To make this more efficient when only one or two of these items change (e.g. highlighting due to mouseover), keep the dirty items in a linked list. Of course, this makes the draw first version of the chart less efficient. There are more fancy ways of implementing the double-linked list, but the few ns gained in the render thread are hardly worth it. Signed-off-by: Berthold Stoeger <bstoeger@mail.tuwien.ac.at>
Diffstat (limited to 'stats/statsview.cpp')
-rw-r--r--stats/statsview.cpp58
1 files changed, 36 insertions, 22 deletions
diff --git a/stats/statsview.cpp b/stats/statsview.cpp
index e893c3ebb..40d23eed3 100644
--- a/stats/statsview.cpp
+++ b/stats/statsview.cpp
@@ -36,9 +36,10 @@ StatsView::StatsView(QQuickItem *parent) : QQuickItem(parent),
xAxis(nullptr),
yAxis(nullptr),
draggedItem(nullptr),
- rootNode(nullptr)
+ rootNode(nullptr),
+ firstDirtyChartItem(nullptr),
+ lastDirtyChartItem(nullptr)
{
- chartItems.reset(new std::vector<ChartItem *>[(size_t)ChartZValue::Count]);
setFlag(ItemHasContents, true);
connect(&diveListNotifier, &DiveListNotifier::numShownChanged, this, &StatsView::replotIfVisible);
@@ -124,11 +125,11 @@ QSGNode *StatsView::updatePaintNode(QSGNode *oldNode, QQuickItem::UpdatePaintNod
plotAreaChanged(plotRect.size());
}
- for (int i = 0; i < (int)ChartZValue::Count; ++i) {
- for (ChartItem *item: chartItems[i]) {
- if (item->dirty)
- item->render();
- }
+ for (ChartItem *item = std::exchange(firstDirtyChartItem, nullptr); item;
+ item = std::exchange(item->dirtyNext, nullptr)) {
+ item->render();
+ item->dirty = false;
+ item->dirtyPrev = nullptr;
}
img->fill(Qt::transparent);
@@ -145,22 +146,34 @@ void StatsView::addQSGNode(QSGNode *node, ChartZValue z)
rootNode->zNodes[idx]->appendChildNode(node);
}
-// Currently this does an inefficient linear search in the chart-item vector.
-// However, we entertain one vector of items per Z-value and currently
-// only the infobox is explicitly deleted, which has a unique Z-value.
-void StatsView::unregisterChartItem(const ChartItem *item)
+void StatsView::unregisterDirtyChartItem(ChartItem &item)
{
- int idx = std::clamp((int)item->zValue, 0, (int)ChartZValue::Count - 1);
- std::vector<ChartItem *> &v = chartItems[idx];
- auto it = std::find(v.begin(), v.end(), item);
- if (it != v.end())
- v.erase(it);
+ if (!item.dirty)
+ return;
+ if (item.dirtyNext)
+ item.dirtyNext->dirtyPrev = item.dirtyPrev;
+ else
+ lastDirtyChartItem = item.dirtyPrev;
+ if (item.dirtyPrev)
+ item.dirtyPrev->dirtyNext = item.dirtyNext;
+ else
+ firstDirtyChartItem = item.dirtyNext;
+ item.dirtyPrev = item.dirtyNext = nullptr;
+ item.dirty = false;
}
-void StatsView::registerChartItem(ChartItem *item)
+void StatsView::registerDirtyChartItem(ChartItem &item)
{
- int idx = std::clamp((int)item->zValue, 0, (int)ChartZValue::Count - 1);
- chartItems[idx].push_back(item);
+ if (item.dirty)
+ return;
+ if (!firstDirtyChartItem) {
+ firstDirtyChartItem = &item;
+ } else {
+ item.dirtyPrev = lastDirtyChartItem;
+ lastDirtyChartItem->dirtyNext = &item;
+ }
+ lastDirtyChartItem = &item;
+ item.dirty = true;
}
QQuickWindow *StatsView::w() const
@@ -326,9 +339,10 @@ void StatsView::reset()
highlightedSeries = nullptr;
xAxis = yAxis = nullptr;
draggedItem = nullptr;
- if (rootNode) {
- for (int i = 0; i < (int)ChartZValue::Count; ++i)
- chartItems[i].clear(); // non-owning pointers
+ for (ChartItem *item = std::exchange(firstDirtyChartItem, nullptr); item;
+ item = std::exchange(item->dirtyNext, nullptr)) {
+ item->dirty = false;
+ item->dirtyPrev = nullptr;
}
legend.reset();
series.clear();