我正在玩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
收到这个非常奇怪的错误通知,但代码编译并运行良好:
为什么呢?......
答案 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
它不再起作用了。任何人都可以解释原因吗?因为wnd
是parent
毕竟。
相反,它包含可通过QQuickItem
函数访问的contentItem()
元素。
parent
属性是指QQuickItem
个对象。
因此,在您的示例中,titleBar.parent
指的是wnd
而不是wnd
本身的元素。
如果内部尝试在调用titleBar.parent
之前动态QWindow*
MyCustomTitleBar::setHostWidget()
,则由于上述(2)而失败(在这种情况下,您应该看到相应的错误)控制台)。