我正在使用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
为什麽?
2条答案
按热度按时间x7yiwoj41#
因为
titleBar.parent
不是wnd
,而是wnd.contentItem
,这就是为什么你发现它的parent
不是wnd
。为什么是
wnd.contentItem
而不是wnd
?一般来说,任何项都将成为它所包含的所有子项的
parent
,但Window
并非如此。这里的问题是,
parent
属性的类型是Item
。可悲的是,Window
并不继承自Item
。因此,Window
应该包含一个真实的Item
,以成为其所有子元素的parent
。这就是Window.contentItem
的用途。92dk7w1h2#
但问题是: 如果我将
hostWidget: wnd
更改为hostWidget: parent
,它就不再工作了。 有人能解释一下为什么吗?因为wnd
毕竟是parent
。**QQuickWindow
。QQuickWindow
不继承自QQuickItem
。1.相反,它包含一个
QQuickItem
元素,可通过其contentItem()
函数访问。parent
属性引用QQuickItem
对象。因此,在你的例子中,
titleBar.parent
指的是wnd
的一个元素,而不是wnd
本身。如果在调用
MyCustomTitleBar::setHostWidget()
之前尝试在内部将titleBar.parent
动态转换为QWindow*
,则会由于上述(2)而失败(在这种情况下,您应该在控制台中看到相应的错误)。