diff options
author | Eduardo M. Fleury <eduardo.fleury@openbossa.org> | 2009-08-08 00:52:46 (GMT) |
---|---|---|
committer | Jan-Arve Sæther <jan-arve.saether@nokia.com> | 2009-08-10 09:00:53 (GMT) |
commit | e2436e25b066ed8236b9cba3483d4a0b87500c91 (patch) | |
tree | 4056ace567943db48969cf60df0c7eca07f84ea8 /src/gui/graphicsview | |
parent | 2f5516cc5a8990e6d1bc5d8d557c421dc0150265 (diff) | |
download | Qt-e2436e25b066ed8236b9cba3483d4a0b87500c91.zip Qt-e2436e25b066ed8236b9cba3483d4a0b87500c91.tar.gz Qt-e2436e25b066ed8236b9cba3483d4a0b87500c91.tar.bz2 |
QGraphicsAnchorLayout: Enable graph simplification
This commit enables graph simplification in its simplest way, which is
to simplify the graph, run simplex and then restore it back.
It also adds code to handle cases where the simplification handles
the whole graph, and that therefore no simplex needs to be run.
Additionally, the updateChildrenSizes() method is called on every
root anchor in order to propagate the new sizes down the tree.
A know issue appeared after this commit, as the quote below explains:
=====
Q_ASSERT error in "Parallel" test
That's something that started happening in the Parallel test
(tests/auto/qgraphicsanchorlayout) after my commit of today, that enables
simplification. I haven't been able to investigate it deeply but that's what
I know about it:
- It's triggered when the layout is destroyed, more precisely, from within
"deleteLayoutEdges". What happens is that the layout structural anchors are
missing at that point.
- I suspect there's something to do with the simplifyGraph() and/or
restoreSimplifyGraph() code. I say that because I tried moving the
"restoreSimplifiedGraph" call from the end of the calculateGraph(s) method
to the line exactly below the "simplifyGraph()" and the error still happens.
That means that simplifying and then immediately restoring the graph might
not leave it to the point where it was before the simplification.
====
Signed-off-by: Eduardo M. Fleury <eduardo.fleury@openbossa.org>
Diffstat (limited to 'src/gui/graphicsview')
-rw-r--r-- | src/gui/graphicsview/qgraphicsanchorlayout_p.cpp | 117 |
1 files changed, 99 insertions, 18 deletions
diff --git a/src/gui/graphicsview/qgraphicsanchorlayout_p.cpp b/src/gui/graphicsview/qgraphicsanchorlayout_p.cpp index 52fac1d..e44b59a 100644 --- a/src/gui/graphicsview/qgraphicsanchorlayout_p.cpp +++ b/src/gui/graphicsview/qgraphicsanchorlayout_p.cpp @@ -197,6 +197,13 @@ static void simplifySequentialChunk(Graph<AnchorVertex, AnchorData> *graph, sequence->minSize = min; sequence->prefSize = pref; sequence->maxSize = max; + + // Unless these values are overhidden by the simplex solver later-on, + // anchors will keep their own preferred size. + sequence->sizeAtMinimum = pref; + sequence->sizeAtPreferred = pref; + sequence->sizeAtMaximum = pref; + sequence->setVertices(vertices); sequence->origin = data->origin == vertices.last() ? before : after; AnchorData *newAnchor = sequence; @@ -208,6 +215,11 @@ static void simplifySequentialChunk(Graph<AnchorVertex, AnchorData> *graph, newAnchor->minSize = min; newAnchor->prefSize = pref; newAnchor->maxSize = max; + + // Same as above, by default, keep preferred size. + newAnchor->sizeAtMinimum = pref; + newAnchor->sizeAtPreferred = pref; + newAnchor->sizeAtMaximum = pref; } graph->createEdge(before, after, newAnchor); } @@ -1018,6 +1030,14 @@ void QGraphicsAnchorLayoutPrivate::calculateGraphs( // Reset the nominal sizes of each anchor based on the current item sizes setAnchorSizeHintsFromDefaults(orientation); + // Simplify the graph + // ### Ideally we would like to do that if, and only if, anchors had + // been added or removed since the last time this method was called. + // However, as the two setAnchorSizeHints methods above are not + // ready to be run on top of a simplified graph, we must simplify + // and restore it every time we get here. + simplifyGraph(orientation); + // Traverse all graph edges and store the possible paths to each vertex findPaths(orientation); @@ -1059,25 +1079,41 @@ void QGraphicsAnchorLayoutPrivate::calculateGraphs( AnchorVertex *v = internalVertex(q, end); GraphPath trunkPath = graphPaths[orientation].value(v); - // Solve min and max size hints for trunk - QPair<qreal, qreal> minMax = solveMinMax(trunkConstraints, trunkPath); - sizeHints[orientation][Qt::MinimumSize] = minMax.first; - sizeHints[orientation][Qt::MaximumSize] = minMax.second; - - // Solve for preferred. The objective function is calculated from the constraints - // and variables internally. - solvePreferred(trunkConstraints); - - // Calculate and set the preferred size for the layout from the edge sizes that - // were calculated above. - qreal pref(0.0); - foreach (const AnchorData *ad, trunkPath.positives) { - pref += ad->sizeAtPreferred; - } - foreach (const AnchorData *ad, trunkPath.negatives) { - pref -= ad->sizeAtPreferred; + if (!trunkConstraints.isEmpty()) { + // Solve min and max size hints for trunk + QPair<qreal, qreal> minMax = solveMinMax(trunkConstraints, trunkPath); + sizeHints[orientation][Qt::MinimumSize] = minMax.first; + sizeHints[orientation][Qt::MaximumSize] = minMax.second; + + // Solve for preferred. The objective function is calculated from the constraints + // and variables internally. + solvePreferred(trunkConstraints); + + // Calculate and set the preferred size for the layout from the edge sizes that + // were calculated above. + qreal pref(0.0); + foreach (const AnchorData *ad, trunkPath.positives) { + pref += ad->sizeAtPreferred; + } + foreach (const AnchorData *ad, trunkPath.negatives) { + pref -= ad->sizeAtPreferred; + } + sizeHints[orientation][Qt::PreferredSize] = pref; + } else { + // No Simplex is necessary because the path was simplified all the way to a single + // anchor. + Q_ASSERT(trunkPath.positives.count() == 1); + Q_ASSERT(trunkPath.negatives.count() == 0); + + AnchorData *ad = trunkPath.positives.toList()[0]; + ad->sizeAtMinimum = ad->minSize; + ad->sizeAtPreferred = ad->prefSize; + ad->sizeAtMaximum = ad->maxSize; + + sizeHints[orientation][Qt::MinimumSize] = ad->sizeAtMinimum; + sizeHints[orientation][Qt::PreferredSize] = ad->sizeAtPreferred; + sizeHints[orientation][Qt::MaximumSize] = ad->sizeAtMaximum; } - sizeHints[orientation][Qt::PreferredSize] = pref; // Delete the constraints, we won't use them anymore. qDeleteAll(sizeHintConstraints); @@ -1116,6 +1152,51 @@ void QGraphicsAnchorLayoutPrivate::calculateGraphs( qDeleteAll(constraints[orientation]); constraints[orientation].clear(); graphPaths[orientation].clear(); // ### + + // Propagate the new sizes down the simplified graph, ie. tell the group anchors + // to set their children anchors sizes. + + // ### Note that we can split the anchors into two categories: + // 1) Those that appeared in at least one constraint and so went through the + // Simplex solver. Either as a Trunk or Non-Trunk variable. + // 2) Those that did not go through the Simplex solver at all. + // + // Anchors of the type (1) had its effective sizes (ie. sizeAtMinimum/Pref/Max) + // properly set by the "solveMinMax" and "solvePreferred" methods. + // + // However, those of type (2), still need to have their effective sizes set, + // in that case, to their own nominal values. + // + // Due to the above reasons, we can't simply iterate on the variables that + // belong to a graph part. We have to iterate through _all_ root anchors + // in graph[orientation]. That's why we collect "allAnchors". We gotta make + // this better somehow. + + // ### Did I say that's ugly? + QSet<AnchorData *> allAnchors; + QQueue<AnchorVertex *> queue; + queue << graph[orientation].rootVertex(); + while (!queue.isEmpty()) { + AnchorVertex *vertex = queue.dequeue(); + QList<AnchorVertex *> adjacentVertices = graph[orientation].adjacentVertices(vertex); + for (int i = 0; i < adjacentVertices.count(); ++i) { + AnchorData *edge = graph[orientation].edgeData(vertex, adjacentVertices[i]); + if (allAnchors.contains(edge)) + continue; + allAnchors << edge; + queue << adjacentVertices[i]; + } + } + + // Ok, now that we have all anchors, actually propagate the sizes down its children. + // Note that for anchors that didn't have its effectiveSizes set yet, we use the + // nominal one instead. + QSet<AnchorData *>::const_iterator iter; + for (iter = allAnchors.constBegin(); iter != allAnchors.constEnd(); ++iter) + (*iter)->updateChildrenSizes(); + + // Restore the graph. See the ### note next to the simplifyGraph() call. + restoreSimplifiedGraph(orientation); } void QGraphicsAnchorLayoutPrivate::setAnchorSizeHintsFromDefaults(Orientation orientation) |