// Copyright (C) 2016 The Qt Company Ltd. // SPDX-License-Identifier: LicenseRef-Qt-Commercial OR LGPL-3.0-only OR GPL-2.0-only OR GPL-3.0-only #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" #include #include 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. */ namespace QtGraphicsAnchorLayout { /*! \internal Represents a vertex (anchorage point) in the internal graph */ struct AnchorVertex { AnchorVertex(QGraphicsLayoutItem *item, Qt::AnchorPoint edge) : m_item(item), m_edge(edge) {} AnchorVertex() : m_item(nullptr), m_edge(Qt::AnchorPoint(0)) {} virtual ~AnchorVertex() = default; #ifdef QT_DEBUG virtual inline QString toString() const; #endif QGraphicsLayoutItem *m_item; Qt::AnchorPoint m_edge; // 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(nullptr), to(nullptr), minSize(0), prefSize(0), maxSize(0), minPrefSize(0), maxPrefSize(0), sizeAtMinimum(0), sizeAtPreferred(0), sizeAtMaximum(0), item(nullptr), graphicsAnchor(nullptr), type(Normal), isLayoutAnchor(false), isCenterAnchor(false), isVertical(false), dependency(Independent) {} virtual ~AnchorData(); virtual void updateChildrenSizes() {} void refreshSizeHints(const QLayoutStyleInfo *styleInfo = nullptr); #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 accessible 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 isVertical : 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 QList &vertices, const QList &edges) : AnchorData(), m_edges(edges) { type = AnchorData::Sequential; isVertical = m_edges.at(0)->isVertical; #ifdef QT_DEBUG name = QString::fromLatin1("%1 -- %2").arg(vertices.first()->toString(), vertices.last()->toString()); #else Q_UNUSED(vertices); #endif } virtual void updateChildrenSizes() override; void calculateSizeHints(); const QList 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; isVertical = first->isVertical; // 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() 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) { } AnchorVertex *m_first; AnchorVertex *m_second; AnchorData *m_removedAnchor; QList m_firstAnchors; QList m_secondAnchors; #ifdef QT_DEBUG inline QString toString() const override { return QString::fromLatin1("(%1, %2)").arg(m_first->toString(), m_second->toString()); } #endif }; #ifdef QT_DEBUG inline QString AnchorVertex::toString() const { using namespace Qt::StringLiterals; if (!m_item) return QString::fromLatin1("NULL_%1").arg(quintptr(this)); QString edge; switch (m_edge) { case Qt::AnchorLeft: edge = "Left"_L1; break; case Qt::AnchorHorizontalCenter: edge = "HorizontalCenter"_L1; break; case Qt::AnchorRight: edge = "Right"_L1; break; case Qt::AnchorTop: edge = "Top"_L1; break; case Qt::AnchorVerticalCenter: edge = "VerticalCenter"_L1; break; case Qt::AnchorBottom: edge = "Bottom"_L1; break; default: edge = "None"_L1; break; } QString itemName; if (m_item->isLayout()) { itemName = "layout"_L1; } else { if (QGraphicsItem *item = m_item->graphicsItem()) { itemName = item->data(0).toString(); } } edge.insert(0, "%1_"_L1); 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; }; } // namespace QtGraphicsAnchorLayout using namespace QtGraphicsAnchorLayout; Q_DECLARE_TYPEINFO(GraphPath, Q_RELOCATABLE_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); static QGraphicsAnchorPrivate *get(QGraphicsAnchor *q) { return q->d_func(); } 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 }; typedef Qt::Orientation Orientation [[deprecated]]; [[deprecated]] static inline constexpr Qt::Orientation Horizontal = Qt::Horizontal; [[deprecated]] static inline constexpr Qt::Orientation Vertical = Qt::Vertical; QGraphicsAnchorLayoutPrivate(); static QGraphicsAnchorLayoutPrivate *get(QGraphicsAnchorLayout *q) { return q ? q->d_func() : nullptr; } static Qt::AnchorPoint oppositeEdge( Qt::AnchorPoint edge); static Qt::Orientation edgeOrientation(Qt::AnchorPoint edge) noexcept; static Qt::AnchorPoint pickEdge(Qt::AnchorPoint edge, Qt::Orientation orientation) { if (orientation == Qt::Vertical && int(edge) <= 2) return (Qt::AnchorPoint)(edge + 3); else if (orientation == Qt::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, Qt::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 = nullptr); // 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(Qt::Orientation orientation); // Simplification bool simplifyGraph(Qt::Orientation orientation); bool simplifyVertices(Qt::Orientation orientation); bool simplifyGraphIteration(Qt::Orientation orientation, bool *feasible); bool replaceVertex(Qt::Orientation orientation, AnchorVertex *oldV, AnchorVertex *newV, const QList &edges); void restoreSimplifiedGraph(Qt::Orientation orientation); void restoreSimplifiedAnchor(AnchorData *edge); void restoreSimplifiedConstraints(ParallelAnchorData *parallel); void restoreVertices(Qt::Orientation orientation); bool calculateTrunk(Qt::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(Qt::Orientation orientation); void findPaths(Qt::Orientation orientation); void constraintsFromPaths(Qt::Orientation orientation); void updateAnchorSizes(Qt::Orientation orientation); QList constraintsFromSizeHints(const QList &anchors); struct GraphParts { QList trunkConstraints; QList nonTrunkConstraints; }; GraphParts getGraphParts(Qt::Orientation orientation); void identifyFloatItems(const QSet &visited, Qt::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(Qt::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(Qt::Orientation orientation); void setupEdgesInterpolation(Qt::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 QHVContainer spacings = {-1, -1}; // Size hints from simplex engine QHVContainer> sizeHints = {{-1, -1, -1}, {-1, -1, -1}}; // Items QList 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 QHVContainer> graph; QHVContainer layoutFirstVertex = {}; QHVContainer layoutCentralVertex = {}; QHVContainer layoutLastVertex = {}; // Combined anchors in order of creation QHVContainer> simplifiedVertices; QHVContainer> anchorsFromSimplifiedVertices; // Graph paths and constraints, for both orientations QHVContainer> graphPaths; QHVContainer> constraints; QHVContainer> itemCenterConstraints; // The interpolation interval and progress based on the current size // as well as the key values (minimum, preferred and maximum) QHVContainer interpolationInterval; QHVContainer interpolationProgress = {-1, -1}; QHVContainer graphHasConflicts = {}; QHVContainer> m_floatItems; #if defined(QT_DEBUG) || defined(QT_BUILD_INTERNAL) QHVContainer lastCalculationUsedSimplex; #endif uint calculateGraphCacheDirty : 1; mutable uint styleInfoDirty : 1; mutable QLayoutStyleInfo cachedStyleInfo; friend class QGraphicsAnchorPrivate; }; QT_END_NAMESPACE #endif