forked from quickshell/quickshell
core/window: expose coordinate mapping functions on QsWindow
This commit is contained in:
parent
6d42d26c79
commit
4a0f6382b0
4 changed files with 215 additions and 2 deletions
|
@ -4,8 +4,10 @@
|
|||
#include <qcoreevent.h>
|
||||
#include <qevent.h>
|
||||
#include <qguiapplication.h>
|
||||
#include <qlogging.h>
|
||||
#include <qnamespace.h>
|
||||
#include <qobject.h>
|
||||
#include <qpoint.h>
|
||||
#include <qqmlcontext.h>
|
||||
#include <qqmlengine.h>
|
||||
#include <qqmlinfo.h>
|
||||
|
@ -461,12 +463,71 @@ QQmlListProperty<QObject> ProxyWindowBase::data() {
|
|||
void ProxyWindowBase::onWidthChanged() { this->mContentItem->setWidth(this->width()); }
|
||||
void ProxyWindowBase::onHeightChanged() { this->mContentItem->setHeight(this->height()); }
|
||||
|
||||
QPointF ProxyWindowBase::itemPosition(QQuickItem* item) const {
|
||||
if (!item) {
|
||||
qCritical() << "Cannot map position of null item.";
|
||||
return {};
|
||||
}
|
||||
|
||||
return this->mContentItem->mapFromItem(item, 0, 0);
|
||||
}
|
||||
|
||||
QRectF ProxyWindowBase::itemRect(QQuickItem* item) const {
|
||||
if (!item) {
|
||||
qCritical() << "Cannot map position of null item.";
|
||||
return {};
|
||||
}
|
||||
|
||||
return this->mContentItem->mapFromItem(item, item->boundingRect());
|
||||
}
|
||||
|
||||
QPointF ProxyWindowBase::mapFromItem(QQuickItem* item, QPointF point) const {
|
||||
if (!item) {
|
||||
qCritical() << "Cannot map position of null item.";
|
||||
return {};
|
||||
}
|
||||
|
||||
return this->mContentItem->mapFromItem(item, point);
|
||||
}
|
||||
|
||||
QPointF ProxyWindowBase::mapFromItem(QQuickItem* item, qreal x, qreal y) const {
|
||||
if (!item) {
|
||||
qCritical() << "Cannot map position of null item.";
|
||||
return {};
|
||||
}
|
||||
|
||||
return this->mContentItem->mapFromItem(item, x, y);
|
||||
}
|
||||
|
||||
QRectF ProxyWindowBase::mapFromItem(QQuickItem* item, QRectF rect) const {
|
||||
if (!item) {
|
||||
qCritical() << "Cannot map position of null item.";
|
||||
return {};
|
||||
}
|
||||
|
||||
return this->mContentItem->mapFromItem(item, rect);
|
||||
}
|
||||
|
||||
QRectF
|
||||
ProxyWindowBase::mapFromItem(QQuickItem* item, qreal x, qreal y, qreal width, qreal height) const {
|
||||
if (!item) {
|
||||
qCritical() << "Cannot map position of null item.";
|
||||
return {};
|
||||
}
|
||||
|
||||
return this->mContentItem->mapFromItem(item, x, y, width, height);
|
||||
}
|
||||
|
||||
ProxyWindowAttached::ProxyWindowAttached(QQuickItem* parent): QsWindowAttached(parent) {
|
||||
this->updateWindow();
|
||||
}
|
||||
|
||||
QObject* ProxyWindowAttached::window() const { return this->mWindow; }
|
||||
QQuickItem* ProxyWindowAttached::contentItem() const { return this->mWindow->contentItem(); }
|
||||
QObject* ProxyWindowAttached::window() const { return this->mWindowInterface; }
|
||||
ProxyWindowBase* ProxyWindowAttached::proxyWindow() const { return this->mWindow; }
|
||||
|
||||
QQuickItem* ProxyWindowAttached::contentItem() const {
|
||||
return this->mWindow ? this->mWindow->contentItem() : nullptr;
|
||||
}
|
||||
|
||||
void ProxyWindowAttached::updateWindow() {
|
||||
auto* window = static_cast<QQuickItem*>(this->parent())->window(); // NOLINT
|
||||
|
@ -481,6 +542,7 @@ void ProxyWindowAttached::updateWindow() {
|
|||
void ProxyWindowAttached::setWindow(ProxyWindowBase* window) {
|
||||
if (window == this->mWindow) return;
|
||||
this->mWindow = window;
|
||||
this->mWindowInterface = window ? qobject_cast<WindowInterface*>(window->parent()) : nullptr;
|
||||
emit this->windowChanged();
|
||||
}
|
||||
|
||||
|
|
|
@ -5,6 +5,7 @@
|
|||
#include <qevent.h>
|
||||
#include <qnamespace.h>
|
||||
#include <qobject.h>
|
||||
#include <qpoint.h>
|
||||
#include <qproperty.h>
|
||||
#include <qqmllist.h>
|
||||
#include <qqmlparserstatus.h>
|
||||
|
@ -13,6 +14,8 @@
|
|||
#include <qsurfaceformat.h>
|
||||
#include <qtmetamacros.h>
|
||||
#include <qtypes.h>
|
||||
#include <qvariant.h>
|
||||
#include <qvectornd.h>
|
||||
#include <qwindow.h>
|
||||
|
||||
#include "../core/qmlscreen.hpp"
|
||||
|
@ -67,6 +70,14 @@ public:
|
|||
void operator=(ProxyWindowBase&) = delete;
|
||||
void operator=(ProxyWindowBase&&) = delete;
|
||||
|
||||
Q_INVOKABLE [[nodiscard]] QPointF itemPosition(QQuickItem* item) const;
|
||||
Q_INVOKABLE [[nodiscard]] QRectF itemRect(QQuickItem* item) const;
|
||||
Q_INVOKABLE [[nodiscard]] QPointF mapFromItem(QQuickItem* item, QPointF point) const;
|
||||
Q_INVOKABLE [[nodiscard]] QPointF mapFromItem(QQuickItem* item, qreal x, qreal y) const;
|
||||
Q_INVOKABLE [[nodiscard]] QRectF mapFromItem(QQuickItem* item, QRectF rect) const;
|
||||
Q_INVOKABLE [[nodiscard]] QRectF
|
||||
mapFromItem(QQuickItem* item, qreal x, qreal y, qreal width, qreal height) const;
|
||||
|
||||
void onReload(QObject* oldInstance) override;
|
||||
void ensureQWindow();
|
||||
void createWindow();
|
||||
|
@ -201,6 +212,7 @@ public:
|
|||
explicit ProxyWindowAttached(QQuickItem* parent);
|
||||
|
||||
[[nodiscard]] QObject* window() const override;
|
||||
[[nodiscard]] ProxyWindowBase* proxyWindow() const override;
|
||||
[[nodiscard]] QQuickItem* contentItem() const override;
|
||||
|
||||
protected:
|
||||
|
@ -208,6 +220,7 @@ protected:
|
|||
|
||||
private:
|
||||
ProxyWindowBase* mWindow = nullptr;
|
||||
WindowInterface* mWindowInterface = nullptr;
|
||||
|
||||
void setWindow(ProxyWindowBase* window);
|
||||
};
|
||||
|
|
|
@ -1,14 +1,96 @@
|
|||
#include "windowinterface.hpp"
|
||||
|
||||
#include <qlogging.h>
|
||||
#include <qobject.h>
|
||||
#include <qquickitem.h>
|
||||
#include <qtypes.h>
|
||||
|
||||
#include "proxywindow.hpp"
|
||||
|
||||
QPointF WindowInterface::itemPosition(QQuickItem* item) const {
|
||||
return this->proxyWindow()->itemPosition(item);
|
||||
}
|
||||
|
||||
QRectF WindowInterface::itemRect(QQuickItem* item) const {
|
||||
return this->proxyWindow()->itemRect(item);
|
||||
}
|
||||
|
||||
QPointF WindowInterface::mapFromItem(QQuickItem* item, QPointF point) const {
|
||||
return this->proxyWindow()->mapFromItem(item, point);
|
||||
}
|
||||
|
||||
QPointF WindowInterface::mapFromItem(QQuickItem* item, qreal x, qreal y) const {
|
||||
return this->proxyWindow()->mapFromItem(item, x, y);
|
||||
}
|
||||
|
||||
QRectF WindowInterface::mapFromItem(QQuickItem* item, QRectF rect) const {
|
||||
return this->proxyWindow()->mapFromItem(item, rect);
|
||||
}
|
||||
|
||||
QRectF
|
||||
WindowInterface::mapFromItem(QQuickItem* item, qreal x, qreal y, qreal width, qreal height) const {
|
||||
return this->proxyWindow()->mapFromItem(item, x, y, width, height);
|
||||
}
|
||||
|
||||
QsWindowAttached::QsWindowAttached(QQuickItem* parent): QObject(parent) {
|
||||
QObject::connect(parent, &QQuickItem::windowChanged, this, &QsWindowAttached::updateWindow);
|
||||
}
|
||||
|
||||
QPointF QsWindowAttached::itemPosition(QQuickItem* item) const {
|
||||
if (auto* proxyWindow = this->proxyWindow()) {
|
||||
return proxyWindow->itemPosition(item);
|
||||
} else {
|
||||
qCritical() << "Cannot call itemPosition before item is a member of a window.";
|
||||
return {};
|
||||
}
|
||||
}
|
||||
|
||||
QRectF QsWindowAttached::itemRect(QQuickItem* item) const {
|
||||
if (auto* proxyWindow = this->proxyWindow()) {
|
||||
return proxyWindow->itemRect(item);
|
||||
} else {
|
||||
qCritical() << "Cannot call itemRect before item is a member of a window.";
|
||||
return {};
|
||||
}
|
||||
}
|
||||
|
||||
QPointF QsWindowAttached::mapFromItem(QQuickItem* item, QPointF point) const {
|
||||
if (auto* proxyWindow = this->proxyWindow()) {
|
||||
return proxyWindow->mapFromItem(item, point);
|
||||
} else {
|
||||
qCritical() << "Cannot call mapFromItem before item is a member of a window.";
|
||||
return {};
|
||||
}
|
||||
}
|
||||
|
||||
QPointF QsWindowAttached::mapFromItem(QQuickItem* item, qreal x, qreal y) const {
|
||||
if (auto* proxyWindow = this->proxyWindow()) {
|
||||
return proxyWindow->mapFromItem(item, x, y);
|
||||
} else {
|
||||
qCritical() << "Cannot call mapFromItem before item is a member of a window.";
|
||||
return {};
|
||||
}
|
||||
}
|
||||
|
||||
QRectF QsWindowAttached::mapFromItem(QQuickItem* item, QRectF rect) const {
|
||||
if (auto* proxyWindow = this->proxyWindow()) {
|
||||
return proxyWindow->mapFromItem(item, rect);
|
||||
} else {
|
||||
qCritical() << "Cannot call mapFromItem before item is a member of a window.";
|
||||
return {};
|
||||
}
|
||||
}
|
||||
|
||||
QRectF
|
||||
QsWindowAttached::mapFromItem(QQuickItem* item, qreal x, qreal y, qreal width, qreal height) const {
|
||||
if (auto* proxyWindow = this->proxyWindow()) {
|
||||
return proxyWindow->mapFromItem(item, x, y, width, height);
|
||||
} else {
|
||||
qCritical() << "Cannot call mapFromItem before item is a member of a window.";
|
||||
return {};
|
||||
}
|
||||
}
|
||||
|
||||
QsWindowAttached* WindowInterface::qmlAttachedProperties(QObject* object) {
|
||||
while (object && !qobject_cast<QQuickItem*>(object)) {
|
||||
object = object->parent();
|
||||
|
|
|
@ -42,6 +42,9 @@ public:
|
|||
/// It provides the following properties
|
||||
/// - `window` - the `QSWindow` object.
|
||||
/// - `contentItem` - the `contentItem` property of the window.
|
||||
///
|
||||
/// @@itemPosition(), @@itemRect(), and @@mapFromItem() can also be called directly
|
||||
/// on the attached object.
|
||||
class WindowInterface: public Reloadable {
|
||||
Q_OBJECT;
|
||||
// clang-format off
|
||||
|
@ -150,6 +153,49 @@ class WindowInterface: public Reloadable {
|
|||
public:
|
||||
explicit WindowInterface(QObject* parent = nullptr): Reloadable(parent) {}
|
||||
|
||||
/// Returns the given Item's position relative to the window. Does not update reactively.
|
||||
///
|
||||
/// Equivalent to calling `window.contentItem.mapFromItem(item, 0, 0)`
|
||||
///
|
||||
/// See also: @@QtQuick.Item.mapFromItem()
|
||||
Q_INVOKABLE [[nodiscard]] QPointF itemPosition(QQuickItem* item) const;
|
||||
/// Returns the given Item's geometry relative to the window. Does not update reactively.
|
||||
///
|
||||
/// Equivalent to calling `window.contentItem.mapFromItem(item, 0, 0, 0, 0)`
|
||||
///
|
||||
/// See also: @@QtQuick.Item.mapFromItem()
|
||||
Q_INVOKABLE [[nodiscard]] QRectF itemRect(QQuickItem* item) const;
|
||||
/// Maps the given point in the coordinate space of `item` to one in the coordinate space
|
||||
/// of this window. Does not update reactively.
|
||||
///
|
||||
/// Equivalent to calling `window.contentItem.mapFromItem(item, point)`
|
||||
///
|
||||
/// See also: @@QtQuick.Item.mapFromItem()
|
||||
Q_INVOKABLE [[nodiscard]] QPointF mapFromItem(QQuickItem* item, QPointF point) const;
|
||||
/// Maps the given point in the coordinate space of `item` to one in the coordinate space
|
||||
/// of this window. Does not update reactively.
|
||||
///
|
||||
/// Equivalent to calling `window.contentItem.mapFromItem(item, x, y)`
|
||||
///
|
||||
/// See also: @@QtQuick.Item.mapFromItem()
|
||||
Q_INVOKABLE [[nodiscard]] QPointF mapFromItem(QQuickItem* item, qreal x, qreal y) const;
|
||||
/// Maps the given rect in the coordinate space of `item` to one in the coordinate space
|
||||
/// of this window. Does not update reactively.
|
||||
///
|
||||
/// Equivalent to calling `window.contentItem.mapFromItem(item, rect)`
|
||||
///
|
||||
/// See also: @@QtQuick.Item.mapFromItem()
|
||||
Q_INVOKABLE [[nodiscard]] QRectF mapFromItem(QQuickItem* item, QRectF rect) const;
|
||||
// clang-format off
|
||||
/// Maps the given rect in the coordinate space of `item` to one in the coordinate space
|
||||
/// of this window. Does not update reactively.
|
||||
///
|
||||
/// Equivalent to calling `window.contentItem.mapFromItem(item, x, y, width, height)`
|
||||
///
|
||||
/// See also: @@QtQuick.Item.mapFromItem()
|
||||
Q_INVOKABLE [[nodiscard]] QRectF mapFromItem(QQuickItem* item, qreal x, qreal y, qreal width, qreal height) const;
|
||||
// clang-format on
|
||||
|
||||
[[nodiscard]] virtual ProxyWindowBase* proxyWindow() const = 0;
|
||||
[[nodiscard]] virtual QQuickItem* contentItem() const = 0;
|
||||
|
||||
|
@ -213,8 +259,18 @@ class QsWindowAttached: public QObject {
|
|||
|
||||
public:
|
||||
[[nodiscard]] virtual QObject* window() const = 0;
|
||||
[[nodiscard]] virtual ProxyWindowBase* proxyWindow() const = 0;
|
||||
[[nodiscard]] virtual QQuickItem* contentItem() const = 0;
|
||||
|
||||
Q_INVOKABLE [[nodiscard]] QPointF itemPosition(QQuickItem* item) const;
|
||||
Q_INVOKABLE [[nodiscard]] QRectF itemRect(QQuickItem* item) const;
|
||||
Q_INVOKABLE [[nodiscard]] QPointF mapFromItem(QQuickItem* item, QPointF point) const;
|
||||
Q_INVOKABLE [[nodiscard]] QPointF mapFromItem(QQuickItem* item, qreal x, qreal y) const;
|
||||
Q_INVOKABLE [[nodiscard]] QRectF mapFromItem(QQuickItem* item, QRectF rect) const;
|
||||
|
||||
Q_INVOKABLE [[nodiscard]] QRectF
|
||||
mapFromItem(QQuickItem* item, qreal x, qreal y, qreal width, qreal height) const;
|
||||
|
||||
signals:
|
||||
void windowChanged();
|
||||
|
||||
|
|
Loading…
Add table
Add a link
Reference in a new issue