Qt快速父级,与父级的id不同

时间:2016-09-10 14:05:13

标签: c++ qt qml qt5 qtquick2

我正在玩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收到这个非常奇怪的错误通知,但代码编译并运行良好: enter image description here

为什么呢?......

2 个答案:

答案 0 :(得分:2)

因为titleBar.parent不是wnd,而是wnd.contentItem。这就是为什么您发现其parent不是wnd

为什么wnd.contentItem代替wnd

一般来说,任何项目都会成为它所包含的所有子项目的parent,但Window并非如此。

这里的问题是parent属性的类型是Item。遗憾的是,Window并未继承Item。因此,Window应该包含真实的Item作为其所有孩子的parent。这就是Window.contentItem的用途。

答案 1 :(得分:1)

  

但问题是:如果我将hostWidget: wnd更改为hostWidget: parent   它不再起作用了。任何人都可以解释原因吗?因为wndparent   毕竟。

  1. Window QML type实例化QQuickWindow

  2. QQuickWindow不会继承QQuickItem

  3. 相反,它包含可通过QQuickItem函数访问的contentItem()元素。

  4. parent属性是指QQuickItem个对象。

  5. 因此,在您的示例中,titleBar.parent指的是wnd而不是wnd本身的元素。

    如果内部尝试在调用titleBar.parent之前动态QWindow* MyCustomTitleBar::setHostWidget(),则由于上述(2)而失败(在这种情况下,您应该看到相应的错误)控制台)。