c++ Qt Quick parent,与父代的'id'不同

taor4pac  于 12个月前  发布在  其他
关注(0)|答案(2)|浏览(87)

我正在使用Qt Quick,我想为我的应用程序创建一个标题栏。所以我继承了QQuickPaintedItem,在上面画了一点,并想把它用作Window的标题栏。我使用的是Qt 5.7。这成功了,但只是在一定程度上;我将在代码后解释更多。
我是这样做的:

main.cpp

#include <QGuiApplication>
#include <QQmlApplicationEngine>

#include "mycustomtitlebar.h"

int main(int argc, char *argv[])
{
    QGuiApplication app(argc, argv);

    qmlRegisterType<MyCustomTitleBar>("my.custom.lib", 1, 0, "MyCustomTitleBar");

    QQmlApplicationEngine engine;
    engine.load(QUrl(QStringLiteral("qrc:/main.qml")));

    return app.exec();
}

字符串

main.qml

import QtQuick 2.7
import QtQuick.Window 2.2
import my.custom.lib 1.0

Window {
    id: wnd
    visible: true
    width: 640
    height: 480
    title: qsTr("Hello World")

    MyCustomTitleBar {
        id: titleBar
        anchors.top: parent.top
        anchors.left: parent.left
        anchors.right: parent.right
        height: 100
        hostWidget: wnd
    }

    Rectangle {
        color: "beige"
        anchors.top: titleBar.bottom
        anchors.left: parent.left
        anchors.right: parent.right
        anchors.bottom: parent.bottom
    }
}

mycustomtitlebar.h

#ifndef MYCUSTOMTITLEBAR_H
#define MYCUSTOMTITLEBAR_H

#include <QQuickPaintedItem>
#include <QPoint>

class MyCustomTitleBar : public QQuickPaintedItem
{
    Q_OBJECT

    Q_PROPERTY(QWindow *hostWidget READ hostWidget WRITE setHostWidget)

public:
    MyCustomTitleBar(QQuickItem *parent = 0);

protected:
    virtual void paint(QPainter *pPainter) Q_DECL_OVERRIDE;
    virtual void mousePressEvent(QMouseEvent *pEvent) Q_DECL_OVERRIDE;
    virtual void mouseMoveEvent(QMouseEvent *pEvent) Q_DECL_OVERRIDE;
    virtual void mouseReleaseEvent(QMouseEvent *pEvent) Q_DECL_OVERRIDE;

private:
    QWindow *hostWidget() const;
    void setHostWidget(QWindow *pHostWidget);

private:
    QWindow *m_pHostWidget;
    QPoint m_initialMousePosition;
    bool m_leftMouseButtonPressed;
};

#endif // MYCUSTOMTITLEBAR_H

mycustomtitlebar.cpp

#include "mycustomtitlebar.h"

#include <QPainter>
#include <QDragMoveEvent>
#include <QWindow>

MyCustomTitleBar::MyCustomTitleBar(QQuickItem *parent)
    : QQuickPaintedItem(parent),
      m_leftMouseButtonPressed(false),
      m_pHostWidget(Q_NULLPTR)
{
    setAcceptedMouseButtons(Qt::AllButtons);
}

void MyCustomTitleBar::paint(QPainter *pPainter)
{
    // Dummy drawing...
    const QRect myRect(10, 10, width() - 20, height() - 20);
    qDebug() << myRect;
    pPainter->drawRect(myRect);
}

void MyCustomTitleBar::mousePressEvent(QMouseEvent *pEvent)
{
    m_leftMouseButtonPressed = true;
    m_initialMousePosition = pEvent->pos();
}

void MyCustomTitleBar::mouseMoveEvent(QMouseEvent *pEvent)
{
    if (m_leftMouseButtonPressed) {
        if (!m_pHostWidget) {
            qDebug() << Q_FUNC_INFO << "Host widget not set. Please set host widget";
            return;
        }

        const QPoint newMousePosition = pEvent->pos() - m_initialMousePosition + m_pHostWidget->position();
        m_pHostWidget->setPosition(newMousePosition);
    }

    QQuickPaintedItem::mouseMoveEvent(pEvent);
}

void MyCustomTitleBar::mouseReleaseEvent(QMouseEvent *pEvent)
{
    m_leftMouseButtonPressed = false;
}

QWindow *MyCustomTitleBar::hostWidget() const
{
    return m_pHostWidget;
}

void MyCustomTitleBar::setHostWidget(QWindow *pHostWidget)
{
    m_pHostWidget = pHostWidget;
}


现在这段代码工作得很好,我运行应用程序,我可以点击标题栏并拖动,整个窗口移动到我想要的地方。

但是,这里有一个问题:如果我把hostWidget: wnd改成hostWidget: parent,它就不工作了.有人能解释一下为什么吗?因为wnd毕竟是parent.

P.S.
我也从Qt Creator得到了这个非常奇怪的错误通知,但代码编译并运行良好:x1c 0d1x
为什麽?

x7yiwoj4

x7yiwoj41#

因为titleBar.parent不是wnd,而是wnd.contentItem,这就是为什么你发现它的parent不是wnd

为什么是wnd.contentItem而不是wnd

一般来说,任何项都将成为它所包含的所有子项的parent,但Window并非如此。
这里的问题是,parent属性的类型是Item。可悲的是,Window并不继承自Item。因此,Window应该包含一个真实的Item,以成为其所有子元素的parent。这就是Window.contentItem的用途。

92dk7w1h

92dk7w1h2#

但问题是: 如果我将hostWidget: wnd更改为hostWidget: parent,它就不再工作了。 有人能解释一下为什么吗?因为wnd毕竟是parent。**

  1. Window QML type示例化QQuickWindow
  2. QQuickWindow不继承自QQuickItem
    1.相反,它包含一个QQuickItem元素,可通过其contentItem()函数访问。
  3. parent属性引用QQuickItem对象。
    因此,在你的例子中,titleBar.parent指的是wnd的一个元素,而不是wnd本身。
    如果在调用MyCustomTitleBar::setHostWidget()之前尝试在内部将titleBar.parent动态转换为QWindow*,则会由于上述(2)而失败(在这种情况下,您应该在控制台中看到相应的错误)。

相关问题