// Copyright (C) 2017 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 #include "qquickpane_p.h" #include "qquickpane_p_p.h" #include "qquickcontentitem_p.h" #include QT_BEGIN_NAMESPACE Q_LOGGING_CATEGORY(lcPane, "qt.quick.controls.pane") /*! \qmltype Pane \inherits Control //! \instantiates QQuickPane \inqmlmodule QtQuick.Controls \since 5.7 \ingroup qtquickcontrols2-containers \ingroup qtquickcontrols2-focusscopes \brief Provides a background matching with the application style and theme. Pane provides a background color that matches with the application style and theme. Pane does not provide a layout of its own, but requires you to position its contents, for instance by creating a \l RowLayout or a \l ColumnLayout. Items declared as children of a Pane are automatically parented to the Pane's \l[QtQuickControls2]{Control::}{contentItem}. Items created dynamically need to be explicitly parented to the \c contentItem. \section1 Content Sizing If only a single item is used within a Pane, it will resize to fit the implicit size of its contained item. This makes it particularly suitable for use together with layouts. \image qtquickcontrols2-pane.png \snippet qtquickcontrols2-pane.qml 1 Sometimes there might be two items within the pane: \code Pane { SwipeView { // ... } PageIndicator { anchors.horizontalCenter: parent.horizontalCenter anchors.bottom: parent.bottom } } \endcode In this case, Pane cannot calculate a sensible implicit size. Since we're anchoring the \l PageIndicator over the \l SwipeView, we can simply set the content size to the view's implicit size: \code Pane { contentWidth: view.implicitWidth contentHeight: view.implicitHeight SwipeView { id: view // ... } PageIndicator { anchors.horizontalCenter: parent.horizontalCenter anchors.bottom: parent.bottom } } \endcode If the \l[QtQuickControls2]{Control::}{contentItem} has no implicit size and only one child, Pane will use the implicit size of that child. For example, in the following code, the Pane assumes the size of the Rectangle: \code Pane { Item { Rectangle { implicitWidth: 200 implicitHeight: 200 color: "salmon" } } } \endcode \sa {Customizing Pane}, {Container Controls}, {Focus Management in Qt Quick Controls}, {Event Handling} */ void QQuickPanePrivate::init() { Q_Q(QQuickPane); q->setFlag(QQuickItem::ItemIsFocusScope); q->setAcceptedMouseButtons(Qt::AllButtons); #if QT_CONFIG(cursor) q->setCursor(Qt::ArrowCursor); #endif connect(q, &QQuickControl::implicitContentWidthChanged, this, &QQuickPanePrivate::updateContentWidth); connect(q, &QQuickControl::implicitContentHeightChanged, this, &QQuickPanePrivate::updateContentHeight); } QList QQuickPanePrivate::contentChildItems() const { if (!contentItem) return QList(); return contentItem->childItems(); } QQuickItem *QQuickPanePrivate::getContentItem() { Q_Q(QQuickPane); if (QQuickItem *item = QQuickControlPrivate::getContentItem()) return item; return new QQuickContentItem(q); } void QQuickPanePrivate::itemImplicitWidthChanged(QQuickItem *item) { QQuickControlPrivate::itemImplicitWidthChanged(item); if (item == firstChild) updateImplicitContentWidth(); } void QQuickPanePrivate::itemImplicitHeightChanged(QQuickItem *item) { QQuickControlPrivate::itemImplicitHeightChanged(item); if (item == firstChild) updateImplicitContentHeight(); } void QQuickPanePrivate::contentChildrenChange() { Q_Q(QQuickPane); QQuickItem *newFirstChild = contentChildItems().value(0); if (newFirstChild != firstChild) { if (firstChild) removeImplicitSizeListener(firstChild); if (newFirstChild) addImplicitSizeListener(newFirstChild); firstChild = newFirstChild; } updateImplicitContentSize(); emit q->contentChildrenChanged(); } qreal QQuickPanePrivate::getContentWidth() const { if (!contentItem) return 0; const qreal cw = contentItem->implicitWidth(); if (!qFuzzyIsNull(cw)) return cw; const auto contentChildren = contentChildItems(); if (contentChildren.size() == 1) return contentChildren.first()->implicitWidth(); return 0; } qreal QQuickPanePrivate::getContentHeight() const { if (!contentItem) return 0; const qreal ch = contentItem->implicitHeight(); if (!qFuzzyIsNull(ch)) return ch; const auto contentChildren = contentChildItems(); if (contentChildren.size() == 1) return contentChildren.first()->implicitHeight(); return 0; } void QQuickPanePrivate::updateContentWidth() { Q_Q(QQuickPane); if (hasContentWidth || qFuzzyCompare(contentWidth, implicitContentWidth)) return; const qreal oldContentWidth = contentWidth; contentWidth = implicitContentWidth; qCDebug(lcPane) << "contentWidth of" << q << "changed to" << contentWidth; q->contentSizeChange(QSizeF(contentWidth, contentHeight), QSizeF(oldContentWidth, contentHeight)); emit q->contentWidthChanged(); } void QQuickPanePrivate::updateContentHeight() { Q_Q(QQuickPane); if (hasContentHeight || qFuzzyCompare(contentHeight, implicitContentHeight)) return; const qreal oldContentHeight = contentHeight; contentHeight = implicitContentHeight; qCDebug(lcPane) << "contentHeight of" << q << "changed to" << contentHeight; q->contentSizeChange(QSizeF(contentWidth, contentHeight), QSizeF(contentWidth, oldContentHeight)); emit q->contentHeightChanged(); } /* A pane needs to be opaque to mouse events, so that events don't get propagated through to controls covered by the pane. */ bool QQuickPanePrivate::handlePress(const QPointF &point, ulong timestamp) { QQuickControlPrivate::handlePress(point, timestamp); return true; } QQuickPane::QQuickPane(QQuickItem *parent) : QQuickControl(*(new QQuickPanePrivate), parent) { Q_D(QQuickPane); d->init(); } QQuickPane::~QQuickPane() { Q_D(QQuickPane); d->removeImplicitSizeListener(d->contentItem); d->removeImplicitSizeListener(d->firstChild); } QQuickPane::QQuickPane(QQuickPanePrivate &dd, QQuickItem *parent) : QQuickControl(dd, parent) { Q_D(QQuickPane); d->init(); } /*! \qmlproperty real QtQuick.Controls::Pane::contentWidth This property holds the content width. It is used for calculating the total implicit width of the pane. For more information, see \l {Content Sizing}. \sa contentHeight */ qreal QQuickPane::contentWidth() const { Q_D(const QQuickPane); return d->contentWidth; } void QQuickPane::setContentWidth(qreal width) { Q_D(QQuickPane); d->hasContentWidth = true; if (qFuzzyCompare(d->contentWidth, width)) return; const qreal oldWidth = d->contentWidth; d->contentWidth = width; contentSizeChange(QSizeF(width, d->contentHeight), QSizeF(oldWidth, d->contentHeight)); emit contentWidthChanged(); } void QQuickPane::resetContentWidth() { Q_D(QQuickPane); if (!d->hasContentWidth) return; d->hasContentHeight = false; d->updateContentWidth(); } /*! \qmlproperty real QtQuick.Controls::Pane::contentHeight This property holds the content height. It is used for calculating the total implicit height of the pane. For more information, see \l {Content Sizing}. \sa contentWidth */ qreal QQuickPane::contentHeight() const { Q_D(const QQuickPane); return d->contentHeight; } void QQuickPane::setContentHeight(qreal height) { Q_D(QQuickPane); d->hasContentHeight = true; if (qFuzzyCompare(d->contentHeight, height)) return; const qreal oldHeight = d->contentHeight; d->contentHeight = height; contentSizeChange(QSizeF(d->contentWidth, height), QSizeF(d->contentWidth, oldHeight)); emit contentHeightChanged(); } void QQuickPane::resetContentHeight() { Q_D(QQuickPane); if (!d->hasContentHeight) return; d->hasContentHeight = false; d->updateContentHeight(); } /*! \qmlproperty list QtQuick.Controls::Pane::contentData \qmldefault This property holds the list of content data. The list contains all objects that have been declared in QML as children of the pane. \note Unlike \c contentChildren, \c contentData does include non-visual QML objects. \sa Item::data, contentChildren */ QQmlListProperty QQuickPanePrivate::contentData() { Q_Q(QQuickPane); return QQmlListProperty(q->contentItem(), nullptr, QQuickItemPrivate::data_append, QQuickItemPrivate::data_count, QQuickItemPrivate::data_at, QQuickItemPrivate::data_clear); } /*! \qmlproperty list QtQuick.Controls::Pane::contentChildren This property holds the list of content children. The list contains all items that have been declared in QML as children of the pane. \note Unlike \c contentData, \c contentChildren does not include non-visual QML objects. \sa Item::children, contentData */ QQmlListProperty QQuickPanePrivate::contentChildren() { Q_Q(QQuickPane); return QQmlListProperty(q->contentItem(), nullptr, QQuickItemPrivate::children_append, QQuickItemPrivate::children_count, QQuickItemPrivate::children_at, QQuickItemPrivate::children_clear); } void QQuickPane::componentComplete() { Q_D(QQuickPane); QQuickControl::componentComplete(); d->updateImplicitContentSize(); } void QQuickPane::contentItemChange(QQuickItem *newItem, QQuickItem *oldItem) { Q_D(QQuickPane); QQuickControl::contentItemChange(newItem, oldItem); if (oldItem) { d->removeImplicitSizeListener(oldItem); QObjectPrivate::disconnect(oldItem, &QQuickItem::childrenChanged, d, &QQuickPanePrivate::contentChildrenChange); } if (newItem) { d->addImplicitSizeListener(newItem); QObjectPrivate::connect(newItem, &QQuickItem::childrenChanged, d, &QQuickPanePrivate::contentChildrenChange); } d->contentChildrenChange(); } void QQuickPane::contentSizeChange(const QSizeF &newSize, const QSizeF &oldSize) { Q_UNUSED(newSize); Q_UNUSED(oldSize); } #if QT_CONFIG(accessibility) QAccessible::Role QQuickPane::accessibleRole() const { return QAccessible::Pane; } #endif QT_END_NAMESPACE #include "moc_qquickpane_p.cpp"