/**************************************************************************** ** ** Copyright (C) 2016 The Qt Company Ltd. ** Contact: https://www.qt.io/licensing/ ** ** This file is part of the QtWidgets module of the Qt Toolkit. ** ** $QT_BEGIN_LICENSE:LGPL$ ** Commercial License Usage ** Licensees holding valid commercial Qt licenses may use this file in ** accordance with the commercial license agreement provided with the ** Software or, alternatively, in accordance with the terms contained in ** a written agreement between you and The Qt Company. For licensing terms ** and conditions see https://www.qt.io/terms-conditions. For further ** information use the contact form at https://www.qt.io/contact-us. ** ** GNU Lesser General Public License Usage ** Alternatively, this file may be used under the terms of the GNU Lesser ** General Public License version 3 as published by the Free Software ** Foundation and appearing in the file LICENSE.LGPL3 included in the ** packaging of this file. Please review the following information to ** ensure the GNU Lesser General Public License version 3 requirements ** will be met: https://www.gnu.org/licenses/lgpl-3.0.html. ** ** GNU General Public License Usage ** Alternatively, this file may be used under the terms of the GNU ** General Public License version 2.0 or (at your option) the GNU General ** Public license version 3 or any later version approved by the KDE Free ** Qt Foundation. The licenses are as published by the Free Software ** Foundation and appearing in the file LICENSE.GPL2 and LICENSE.GPL3 ** included in the packaging of this file. Please review the following ** information to ensure the GNU General Public License requirements will ** be met: https://www.gnu.org/licenses/gpl-2.0.html and ** https://www.gnu.org/licenses/gpl-3.0.html. ** ** $QT_END_LICENSE$ ** ****************************************************************************/ #ifndef QGRAPHICSANCHORLAYOUT_P_H #define QGRAPHICSANCHORLAYOUT_P_H // // W A R N I N G // ------------- // // This file is not part of the Qt API. It exists purely as an // implementation detail. This header file may change from version to // version without notice, or even be removed. // // We mean it. // #include #include #include #include "qgraphicslayout_p.h" #include "qgraphicsanchorlayout.h" #include "qgraph_p.h" #include "qsimplex_p.h" QT_REQUIRE_CONFIG(graphicsview); QT_BEGIN_NAMESPACE /* The public QGraphicsAnchorLayout interface represents an anchorage point as a pair of a and a . Internally though, it has a graph of anchorage points (vertices) and anchors (edges), represented by the AnchorVertex and AnchorData structs respectively. */ /*! \internal Represents a vertex (anchorage point) in the internal graph */ struct AnchorVertex { enum Type { Normal = 0, Pair }; AnchorVertex(QGraphicsLayoutItem *item, Qt::AnchorPoint edge) : m_item(item), m_edge(edge), m_type(Normal) {} AnchorVertex() : m_item(0), m_edge(Qt::AnchorPoint(0)), m_type(Normal) {} #ifdef QT_DEBUG inline QString toString() const; #endif QGraphicsLayoutItem *m_item; Qt::AnchorPoint m_edge; uint m_type : 1; // Current distance from this vertex to the layout edge (Left or Top) // Value is calculated from the current anchors sizes. qreal distance; }; /*! \internal Represents an edge (anchor) in the internal graph. */ struct AnchorData : public QSimplexVariable { enum Type { Normal = 0, Sequential, Parallel }; enum Dependency { Independent = 0, Master, Slave }; AnchorData() : QSimplexVariable(), from(0), to(0), minSize(0), prefSize(0), maxSize(0), minPrefSize(0), maxPrefSize(0), sizeAtMinimum(0), sizeAtPreferred(0), sizeAtMaximum(0), item(0), graphicsAnchor(0), type(Normal), isLayoutAnchor(false), isCenterAnchor(false), orientation(0), dependency(Independent) {} virtual ~AnchorData(); virtual void updateChildrenSizes() {} void refreshSizeHints(const QLayoutStyleInfo *styleInfo = 0); #ifdef QT_DEBUG void dump(int indent = 2); inline QString toString() const; QString name; #endif // Anchor is semantically directed AnchorVertex *from; AnchorVertex *to; // Nominal sizes // These are the intrinsic size restrictions for a given item. They are // used as input for the calculation of the actual sizes. // These values are filled by the refreshSizeHints method, based on the // anchor size policy, the size hints of the item it (possibly) represents // and the layout spacing information. qreal minSize; qreal prefSize; qreal maxSize; qreal minPrefSize; qreal maxPrefSize; // Calculated sizes // These attributes define which sizes should that anchor be in when the // layout is at its minimum, preferred or maximum sizes. Values are // calculated by the Simplex solver based on the current layout setup. qreal sizeAtMinimum; qreal sizeAtPreferred; qreal sizeAtMaximum; // References to the classes that represent this anchor in the public world // An anchor may represent a LayoutItem, it may also be acessible externally // through a GraphicsAnchor "handler". QGraphicsLayoutItem *item; QGraphicsAnchor *graphicsAnchor; uint type : 2; // either Normal, Sequential or Parallel uint isLayoutAnchor : 1; // if this anchor is an internal layout anchor uint isCenterAnchor : 1; uint orientation : 1; uint dependency : 2; // either Independent, Master or Slave }; #ifdef QT_DEBUG inline QString AnchorData::toString() const { return QString::fromLatin1("Anchor(%1)").arg(name); } #endif struct SequentialAnchorData : public AnchorData { SequentialAnchorData(const QVector &vertices, const QVector &edges) : AnchorData(), m_children(vertices), m_edges(edges) { type = AnchorData::Sequential; orientation = m_edges.at(0)->orientation; #ifdef QT_DEBUG name = QString::fromLatin1("%1 -- %2").arg(vertices.first()->toString(), vertices.last()->toString()); #endif } virtual void updateChildrenSizes() Q_DECL_OVERRIDE; void calculateSizeHints(); QVector m_children; // list of vertices in the sequence QVector m_edges; // keep the list of edges too. }; struct ParallelAnchorData : public AnchorData { ParallelAnchorData(AnchorData *first, AnchorData *second) : AnchorData(), firstEdge(first), secondEdge(second) { type = AnchorData::Parallel; orientation = first->orientation; // This assert whether the child anchors share their vertices Q_ASSERT(((first->from == second->from) && (first->to == second->to)) || ((first->from == second->to) && (first->to == second->from))); // Our convention will be that the parallel group anchor will have the same // direction as the first anchor. from = first->from; to = first->to; #ifdef QT_DEBUG name = QString::fromLatin1("%1 | %2").arg(first->toString(), second->toString()); #endif } virtual void updateChildrenSizes() Q_DECL_OVERRIDE; bool calculateSizeHints(); bool secondForward() const { // We have the convention that the first children will define the direction of the // pararell group. Note that we can't rely on 'this->from' or 'this->to' because they // might be changed by vertex simplification. return firstEdge->from == secondEdge->from; } AnchorData* firstEdge; AnchorData* secondEdge; QList m_firstConstraints; QList m_secondConstraints; }; struct AnchorVertexPair : public AnchorVertex { AnchorVertexPair(AnchorVertex *v1, AnchorVertex *v2, AnchorData *data) : AnchorVertex(), m_first(v1), m_second(v2), m_removedAnchor(data) { m_type = AnchorVertex::Pair; } AnchorVertex *m_first; AnchorVertex *m_second; AnchorData *m_removedAnchor; QList m_firstAnchors; QList m_secondAnchors; }; #ifdef QT_DEBUG inline QString AnchorVertex::toString() const { if (m_type == Pair) { const AnchorVertexPair *vp = static_cast(this); return QString::fromLatin1("(%1, %2)").arg(vp->m_first->toString(), vp->m_second->toString()); } else if (!m_item) { return QString::fromLatin1("NULL_%1").arg(quintptr(this)); } QString edge; switch (m_edge) { case Qt::AnchorLeft: edge = QLatin1String("Left"); break; case Qt::AnchorHorizontalCenter: edge = QLatin1String("HorizontalCenter"); break; case Qt::AnchorRight: edge = QLatin1String("Right"); break; case Qt::AnchorTop: edge = QLatin1String("Top"); break; case Qt::AnchorVerticalCenter: edge = QLatin1String("VerticalCenter"); break; case Qt::AnchorBottom: edge = QLatin1String("Bottom"); break; default: edge = QLatin1String("None"); break; } QString itemName; if (m_item->isLayout()) { itemName = QLatin1String("layout"); } else { if (QGraphicsItem *item = m_item->graphicsItem()) { itemName = item->data(0).toString(); } } edge.insert(0, QLatin1String("%1_")); return edge.arg(itemName); } #endif /*! \internal Representation of a valid path for a given vertex in the graph. In this struct, "positives" is the set of anchors that have been traversed in the forward direction, while "negatives" is the set with the ones walked backwards. This paths are compared against each other to produce LP Constraints, the exact order in which the anchors were traversed is not relevant. */ class GraphPath { public: GraphPath() {} QSimplexConstraint *constraint(const GraphPath &path) const; #ifdef QT_DEBUG QString toString() const; #endif QSet positives; QSet negatives; }; Q_DECLARE_TYPEINFO(GraphPath, Q_MOVABLE_TYPE); class QGraphicsAnchorLayoutPrivate; /*! \internal */ class QGraphicsAnchorPrivate : public QObjectPrivate { Q_DECLARE_PUBLIC(QGraphicsAnchor) public: explicit QGraphicsAnchorPrivate(int version = QObjectPrivateVersion); ~QGraphicsAnchorPrivate(); void setSpacing(qreal value); void unsetSpacing(); qreal spacing() const; void setSizePolicy(QSizePolicy::Policy policy); QGraphicsAnchorLayoutPrivate *layoutPrivate; AnchorData *data; // Size information for user controlled anchor QSizePolicy::Policy sizePolicy; qreal preferredSize; uint hasSize : 1; // if false, get size from style. }; /*! \internal QGraphicsAnchorLayout private methods and attributes. */ class Q_AUTOTEST_EXPORT QGraphicsAnchorLayoutPrivate : public QGraphicsLayoutPrivate { Q_DECLARE_PUBLIC(QGraphicsAnchorLayout) public: // When the layout geometry is different from its Minimum, Preferred // or Maximum values, interpolation is used to calculate the geometries // of the items. // // Interval represents which interpolation interval are we operating in. enum Interval { MinimumToMinPreferred = 0, MinPreferredToPreferred, PreferredToMaxPreferred, MaxPreferredToMaximum }; // Several structures internal to the layout are duplicated to handle // both Horizontal and Vertical restrictions. // // Orientation is used to reference the right structure in each context enum Orientation { Horizontal = 0, Vertical, NOrientations }; QGraphicsAnchorLayoutPrivate(); static QGraphicsAnchorLayoutPrivate *get(QGraphicsAnchorLayout *q) { return q ? q->d_func() : 0; } static Qt::AnchorPoint oppositeEdge( Qt::AnchorPoint edge); static Orientation edgeOrientation(Qt::AnchorPoint edge); static Qt::AnchorPoint pickEdge(Qt::AnchorPoint edge, Orientation orientation) { if (orientation == Vertical && int(edge) <= 2) return (Qt::AnchorPoint)(edge + 3); else if (orientation == Horizontal && int(edge) >= 3) { return (Qt::AnchorPoint)(edge - 3); } return edge; } // Init methods void createLayoutEdges(); void deleteLayoutEdges(); void createItemEdges(QGraphicsLayoutItem *item); void createCenterAnchors(QGraphicsLayoutItem *item, Qt::AnchorPoint centerEdge); void removeCenterAnchors(QGraphicsLayoutItem *item, Qt::AnchorPoint centerEdge, bool substitute = true); void removeCenterConstraints(QGraphicsLayoutItem *item, Orientation orientation); QGraphicsAnchor *acquireGraphicsAnchor(AnchorData *data) { Q_Q(QGraphicsAnchorLayout); if (!data->graphicsAnchor) { data->graphicsAnchor = new QGraphicsAnchor(q); data->graphicsAnchor->d_func()->data = data; } return data->graphicsAnchor; } // function used by the 4 API functions QGraphicsAnchor *addAnchor(QGraphicsLayoutItem *firstItem, Qt::AnchorPoint firstEdge, QGraphicsLayoutItem *secondItem, Qt::AnchorPoint secondEdge, qreal *spacing = 0); // Helper for Anchor Manipulation methods void addAnchor_helper(QGraphicsLayoutItem *firstItem, Qt::AnchorPoint firstEdge, QGraphicsLayoutItem *secondItem, Qt::AnchorPoint secondEdge, AnchorData *data); QGraphicsAnchor *getAnchor(QGraphicsLayoutItem *firstItem, Qt::AnchorPoint firstEdge, QGraphicsLayoutItem *secondItem, Qt::AnchorPoint secondEdge); void removeAnchor(AnchorVertex *firstVertex, AnchorVertex *secondVertex); void removeAnchor_helper(AnchorVertex *v1, AnchorVertex *v2); void removeAnchors(QGraphicsLayoutItem *item); void removeVertex(QGraphicsLayoutItem *item, Qt::AnchorPoint edge); void correctEdgeDirection(QGraphicsLayoutItem *&firstItem, Qt::AnchorPoint &firstEdge, QGraphicsLayoutItem *&secondItem, Qt::AnchorPoint &secondEdge); QLayoutStyleInfo &styleInfo() const; AnchorData *addAnchorMaybeParallel(AnchorData *newAnchor, bool *feasible); // Activation void calculateGraphs(); void calculateGraphs(Orientation orientation); // Simplification bool simplifyGraph(Orientation orientation); bool simplifyVertices(Orientation orientation); bool simplifyGraphIteration(Orientation orientation, bool *feasible); bool replaceVertex(Orientation orientation, AnchorVertex *oldV, AnchorVertex *newV, const QList &edges); void restoreSimplifiedGraph(Orientation orientation); void restoreSimplifiedAnchor(AnchorData *edge); void restoreSimplifiedConstraints(ParallelAnchorData *parallel); void restoreVertices(Orientation orientation); bool calculateTrunk(Orientation orientation, const GraphPath &trunkPath, const QList &constraints, const QList &variables); bool calculateNonTrunk(const QList &constraints, const QList &variables); // Support functions for calculateGraph() void refreshAllSizeHints(Orientation orientation); void findPaths(Orientation orientation); void constraintsFromPaths(Orientation orientation); void updateAnchorSizes(Orientation orientation); QList constraintsFromSizeHints(const QList &anchors); struct GraphParts { QList trunkConstraints; QList nonTrunkConstraints; }; GraphParts getGraphParts(Orientation orientation); void identifyFloatItems(const QSet &visited, Orientation orientation); void identifyNonFloatItems_helper(const AnchorData *ad, QSet *nonFloatingItemsIdentifiedSoFar); inline AnchorVertex *internalVertex(const QPair &itemEdge) const { return m_vertexList.value(itemEdge).first; } inline AnchorVertex *internalVertex(const QGraphicsLayoutItem *item, Qt::AnchorPoint edge) const { return internalVertex(qMakePair(const_cast(item), edge)); } inline void changeLayoutVertex(Orientation orientation, AnchorVertex *oldV, AnchorVertex *newV) { if (layoutFirstVertex[orientation] == oldV) layoutFirstVertex[orientation] = newV; else if (layoutCentralVertex[orientation] == oldV) layoutCentralVertex[orientation] = newV; else if (layoutLastVertex[orientation] == oldV) layoutLastVertex[orientation] = newV; } AnchorVertex *addInternalVertex(QGraphicsLayoutItem *item, Qt::AnchorPoint edge); void removeInternalVertex(QGraphicsLayoutItem *item, Qt::AnchorPoint edge); // Geometry interpolation methods void setItemsGeometries(const QRectF &geom); void calculateVertexPositions(Orientation orientation); void setupEdgesInterpolation(Orientation orientation); void interpolateEdge(AnchorVertex *base, AnchorData *edge); // Linear Programming solver methods bool solveMinMax(const QList &constraints, const GraphPath &path, qreal *min, qreal *max); bool solvePreferred(const QList &constraints, const QList &variables); bool hasConflicts() const; #ifdef QT_DEBUG void dumpGraph(const QString &name = QString()); #endif qreal spacings[NOrientations]; // Size hints from simplex engine qreal sizeHints[2][3]; // Items QVector items; // Mapping between high level anchorage points (Item, Edge) to low level // ones (Graph Vertices) QHash, QPair > m_vertexList; // Internal graph of anchorage points and anchors, for both orientations Graph graph[2]; AnchorVertex *layoutFirstVertex[2]; AnchorVertex *layoutCentralVertex[2]; AnchorVertex *layoutLastVertex[2]; // Combined anchors in order of creation QList simplifiedVertices[2]; QList anchorsFromSimplifiedVertices[2]; // Graph paths and constraints, for both orientations QMultiHash graphPaths[2]; QList constraints[2]; QList itemCenterConstraints[2]; // The interpolation interval and progress based on the current size // as well as the key values (minimum, preferred and maximum) Interval interpolationInterval[2]; qreal interpolationProgress[2]; bool graphHasConflicts[2]; QSet m_floatItems[2]; #if defined(QT_DEBUG) || defined(QT_BUILD_INTERNAL) bool lastCalculationUsedSimplex[2]; #endif uint calculateGraphCacheDirty : 1; mutable uint styleInfoDirty : 1; mutable QLayoutStyleInfo cachedStyleInfo; friend class QGraphicsAnchorPrivate; }; QT_END_NAMESPACE #endif