forked from quickshell/quickshell
		
	feat: add clickthrough mask to windows
This commit is contained in:
		
							parent
							
								
									5f75c40b67
								
							
						
					
					
						commit
						82aa7d45d3
					
				
					 7 changed files with 329 additions and 2 deletions
				
			
		| 
						 | 
					@ -38,6 +38,7 @@ qt_add_executable(quickshell
 | 
				
			||||||
	src/cpp/qmlglobal.cpp
 | 
						src/cpp/qmlglobal.cpp
 | 
				
			||||||
	src/cpp/qmlscreen.cpp
 | 
						src/cpp/qmlscreen.cpp
 | 
				
			||||||
	src/cpp/watcher.cpp
 | 
						src/cpp/watcher.cpp
 | 
				
			||||||
 | 
						src/cpp/region.cpp
 | 
				
			||||||
)
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
qt_add_qml_module(quickshell URI QuickShell)
 | 
					qt_add_qml_module(quickshell URI QuickShell)
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
							
								
								
									
										2
									
								
								docs
									
										
									
									
									
								
							
							
						
						
									
										2
									
								
								docs
									
										
									
									
									
								
							| 
						 | 
					@ -1 +1 @@
 | 
				
			||||||
Subproject commit 27b3274027251ebf382e31546ef2b350ae2f7b0e
 | 
					Subproject commit 94f07543939dfe682bb382f6802cbe9ff3eea061
 | 
				
			||||||
| 
						 | 
					@ -8,6 +8,7 @@ headers = [
 | 
				
			||||||
	"variants.hpp",
 | 
						"variants.hpp",
 | 
				
			||||||
	"proxywindow.hpp",
 | 
						"proxywindow.hpp",
 | 
				
			||||||
	"layershell.hpp",
 | 
						"layershell.hpp",
 | 
				
			||||||
 | 
						"region.hpp",
 | 
				
			||||||
]
 | 
					]
 | 
				
			||||||
-----
 | 
					-----
 | 
				
			||||||
The core types provided by QuickShell
 | 
					The core types provided by QuickShell
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -4,9 +4,13 @@
 | 
				
			||||||
#include <qqmllist.h>
 | 
					#include <qqmllist.h>
 | 
				
			||||||
#include <qquickitem.h>
 | 
					#include <qquickitem.h>
 | 
				
			||||||
#include <qquickwindow.h>
 | 
					#include <qquickwindow.h>
 | 
				
			||||||
 | 
					#include <qregion.h>
 | 
				
			||||||
 | 
					#include <qtmetamacros.h>
 | 
				
			||||||
#include <qtypes.h>
 | 
					#include <qtypes.h>
 | 
				
			||||||
#include <qwindow.h>
 | 
					#include <qwindow.h>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#include "region.hpp"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
ProxyWindowBase::~ProxyWindowBase() {
 | 
					ProxyWindowBase::~ProxyWindowBase() {
 | 
				
			||||||
	if (this->window != nullptr) {
 | 
						if (this->window != nullptr) {
 | 
				
			||||||
		this->window->deleteLater();
 | 
							this->window->deleteLater();
 | 
				
			||||||
| 
						 | 
					@ -22,11 +26,17 @@ void ProxyWindowBase::earlyInit(QObject* old) {
 | 
				
			||||||
		this->window = oldpw->disownWindow();
 | 
							this->window = oldpw->disownWindow();
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						this->window->setMask(QRegion());
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	// clang-format off
 | 
						// clang-format off
 | 
				
			||||||
	QObject::connect(this->window, &QWindow::visibilityChanged, this, &ProxyWindowBase::visibleChanged);
 | 
						QObject::connect(this->window, &QWindow::visibilityChanged, this, &ProxyWindowBase::visibleChanged);
 | 
				
			||||||
	QObject::connect(this->window, &QWindow::widthChanged, this, &ProxyWindowBase::widthChanged);
 | 
						QObject::connect(this->window, &QWindow::widthChanged, this, &ProxyWindowBase::widthChanged);
 | 
				
			||||||
	QObject::connect(this->window, &QWindow::heightChanged, this, &ProxyWindowBase::heightChanged);
 | 
						QObject::connect(this->window, &QWindow::heightChanged, this, &ProxyWindowBase::heightChanged);
 | 
				
			||||||
	QObject::connect(this->window, &QQuickWindow::colorChanged, this, &ProxyWindowBase::colorChanged);
 | 
						QObject::connect(this->window, &QQuickWindow::colorChanged, this, &ProxyWindowBase::colorChanged);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						QObject::connect(this, &ProxyWindowBase::maskChanged, this, &ProxyWindowBase::onMaskChanged);
 | 
				
			||||||
 | 
						QObject::connect(this, &ProxyWindowBase::widthChanged, this, &ProxyWindowBase::onMaskChanged);
 | 
				
			||||||
 | 
						QObject::connect(this, &ProxyWindowBase::heightChanged, this, &ProxyWindowBase::onMaskChanged);
 | 
				
			||||||
	// clang-format on
 | 
						// clang-format on
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -55,6 +65,36 @@ PROXYPROP(qint32, width, setWidth);
 | 
				
			||||||
PROXYPROP(qint32, height, setHeight);
 | 
					PROXYPROP(qint32, height, setHeight);
 | 
				
			||||||
PROXYPROP(QColor, color, setColor);
 | 
					PROXYPROP(QColor, color, setColor);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					PendingRegion* ProxyWindowBase::mask() { return this->mMask; }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					void ProxyWindowBase::setMask(PendingRegion* mask) {
 | 
				
			||||||
 | 
						if (this->mMask != nullptr) {
 | 
				
			||||||
 | 
							this->mMask->deleteLater();
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if (mask != nullptr) {
 | 
				
			||||||
 | 
							mask->setParent(this);
 | 
				
			||||||
 | 
							this->mMask = mask;
 | 
				
			||||||
 | 
							QObject::connect(mask, &PendingRegion::changed, this, &ProxyWindowBase::maskChanged);
 | 
				
			||||||
 | 
							emit this->maskChanged();
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					void ProxyWindowBase::onMaskChanged() {
 | 
				
			||||||
 | 
						QRegion mask;
 | 
				
			||||||
 | 
						if (this->mMask != nullptr) {
 | 
				
			||||||
 | 
							// if left as the default, dont combine it with the whole window area, leave it as is.
 | 
				
			||||||
 | 
							if (this->mMask->mIntersection == Intersection::Combine) {
 | 
				
			||||||
 | 
								mask = this->mMask->build();
 | 
				
			||||||
 | 
							} else {
 | 
				
			||||||
 | 
								auto windowRegion = QRegion(QRect(0, 0, this->width(), this->height()));
 | 
				
			||||||
 | 
								mask = this->mMask->applyTo(windowRegion);
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						this->window->setMask(mask);
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
// see:
 | 
					// see:
 | 
				
			||||||
// https://code.qt.io/cgit/qt/qtdeclarative.git/tree/src/quick/items/qquickwindow.cpp
 | 
					// https://code.qt.io/cgit/qt/qtdeclarative.git/tree/src/quick/items/qquickwindow.cpp
 | 
				
			||||||
// https://code.qt.io/cgit/qt/qtdeclarative.git/tree/src/quick/items/qquickitem.cpp
 | 
					// https://code.qt.io/cgit/qt/qtdeclarative.git/tree/src/quick/items/qquickitem.cpp
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -10,6 +10,7 @@
 | 
				
			||||||
#include <qtmetamacros.h>
 | 
					#include <qtmetamacros.h>
 | 
				
			||||||
#include <qtypes.h>
 | 
					#include <qtypes.h>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#include "region.hpp"
 | 
				
			||||||
#include "scavenge.hpp"
 | 
					#include "scavenge.hpp"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
// Proxy to an actual window exposing a limited property set with the ability to
 | 
					// Proxy to an actual window exposing a limited property set with the ability to
 | 
				
			||||||
| 
						 | 
					@ -33,7 +34,7 @@ class ProxyWindowBase: public Scavenger {
 | 
				
			||||||
	/// The visibility of the window.
 | 
						/// The visibility of the window.
 | 
				
			||||||
	///
 | 
						///
 | 
				
			||||||
	/// > [!INFO] Windows are not visible by default so you will need to set this to make the window
 | 
						/// > [!INFO] Windows are not visible by default so you will need to set this to make the window
 | 
				
			||||||
	/// appear.
 | 
						/// > appear.
 | 
				
			||||||
	Q_PROPERTY(bool visible READ isVisible WRITE setVisible NOTIFY visibleChanged);
 | 
						Q_PROPERTY(bool visible READ isVisible WRITE setVisible NOTIFY visibleChanged);
 | 
				
			||||||
	Q_PROPERTY(qint32 width READ width WRITE setWidth NOTIFY widthChanged);
 | 
						Q_PROPERTY(qint32 width READ width WRITE setWidth NOTIFY widthChanged);
 | 
				
			||||||
	Q_PROPERTY(qint32 height READ height WRITE setHeight NOTIFY heightChanged);
 | 
						Q_PROPERTY(qint32 height READ height WRITE setHeight NOTIFY heightChanged);
 | 
				
			||||||
| 
						 | 
					@ -52,6 +53,49 @@ class ProxyWindowBase: public Scavenger {
 | 
				
			||||||
	/// > }
 | 
						/// > }
 | 
				
			||||||
	/// > ```
 | 
						/// > ```
 | 
				
			||||||
	Q_PROPERTY(QColor color READ color WRITE setColor NOTIFY colorChanged);
 | 
						Q_PROPERTY(QColor color READ color WRITE setColor NOTIFY colorChanged);
 | 
				
			||||||
 | 
						/// The clickthrough mask. Defaults to null.
 | 
				
			||||||
 | 
						///
 | 
				
			||||||
 | 
						/// If non null then the clickable areas of the window will be determined by the provided region.
 | 
				
			||||||
 | 
						///
 | 
				
			||||||
 | 
						/// ```qml
 | 
				
			||||||
 | 
						/// ProxyShellWindow {
 | 
				
			||||||
 | 
						///   // The mask region is set to `rect`, meaning only `rect` is clickable.
 | 
				
			||||||
 | 
						///   // All other clicks pass through the window to ones behind it.
 | 
				
			||||||
 | 
						///   mask: Region { item: rect }
 | 
				
			||||||
 | 
						///
 | 
				
			||||||
 | 
						///   Rectangle {
 | 
				
			||||||
 | 
						///     id: rect
 | 
				
			||||||
 | 
						///
 | 
				
			||||||
 | 
						///     anchors.centerIn: parent
 | 
				
			||||||
 | 
						///     width: 100
 | 
				
			||||||
 | 
						///     height: 100
 | 
				
			||||||
 | 
						///   }
 | 
				
			||||||
 | 
						/// }
 | 
				
			||||||
 | 
						/// ```
 | 
				
			||||||
 | 
						///
 | 
				
			||||||
 | 
						/// If the provided region's intersection mode is `Combine` (the default),
 | 
				
			||||||
 | 
						/// then the region will be used as is. Otherwise it will be applied on top of the window region.
 | 
				
			||||||
 | 
						///
 | 
				
			||||||
 | 
						/// For example, setting the intersection mode to `Xor` will invert the mask and make everything in
 | 
				
			||||||
 | 
						/// the mask region not clickable and pass through clicks inside it through the window.
 | 
				
			||||||
 | 
						///
 | 
				
			||||||
 | 
						/// ```qml
 | 
				
			||||||
 | 
						/// ProxyShellWindow {
 | 
				
			||||||
 | 
						///   // The mask region is set to `rect`, but the intersection mode is set to `Xor`.
 | 
				
			||||||
 | 
						///   // This inverts the mask causing all clicks inside `rect` to be passed to the window
 | 
				
			||||||
 | 
						///   // behind this one.
 | 
				
			||||||
 | 
						///   mask: Region { item: rect; intersection: Intersection.Xor }
 | 
				
			||||||
 | 
						///
 | 
				
			||||||
 | 
						///   Rectangle {
 | 
				
			||||||
 | 
						///     id: rect
 | 
				
			||||||
 | 
						///
 | 
				
			||||||
 | 
						///     anchors.centerIn: parent
 | 
				
			||||||
 | 
						///     width: 100
 | 
				
			||||||
 | 
						///     height: 100
 | 
				
			||||||
 | 
						///   }
 | 
				
			||||||
 | 
						/// }
 | 
				
			||||||
 | 
						/// ```
 | 
				
			||||||
 | 
						Q_PROPERTY(PendingRegion* mask READ mask WRITE setMask NOTIFY maskChanged);
 | 
				
			||||||
	Q_PROPERTY(QQmlListProperty<QObject> data READ data);
 | 
						Q_PROPERTY(QQmlListProperty<QObject> data READ data);
 | 
				
			||||||
	Q_CLASSINFO("DefaultProperty", "data");
 | 
						Q_CLASSINFO("DefaultProperty", "data");
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -86,6 +130,9 @@ public:
 | 
				
			||||||
	QColor color();
 | 
						QColor color();
 | 
				
			||||||
	void setColor(QColor value);
 | 
						void setColor(QColor value);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						PendingRegion* mask();
 | 
				
			||||||
 | 
						void setMask(PendingRegion* mask);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	QQmlListProperty<QObject> data();
 | 
						QQmlListProperty<QObject> data();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
signals:
 | 
					signals:
 | 
				
			||||||
| 
						 | 
					@ -93,6 +140,10 @@ signals:
 | 
				
			||||||
	void widthChanged(qint32 width);
 | 
						void widthChanged(qint32 width);
 | 
				
			||||||
	void heightChanged(qint32 width);
 | 
						void heightChanged(qint32 width);
 | 
				
			||||||
	void colorChanged(QColor color);
 | 
						void colorChanged(QColor color);
 | 
				
			||||||
 | 
						void maskChanged();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					private slots:
 | 
				
			||||||
 | 
						void onMaskChanged();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
private:
 | 
					private:
 | 
				
			||||||
	static QQmlListProperty<QObject> dataBacker(QQmlListProperty<QObject>* prop);
 | 
						static QQmlListProperty<QObject> dataBacker(QQmlListProperty<QObject>* prop);
 | 
				
			||||||
| 
						 | 
					@ -102,6 +153,8 @@ private:
 | 
				
			||||||
	static void dataClear(QQmlListProperty<QObject>* prop);
 | 
						static void dataClear(QQmlListProperty<QObject>* prop);
 | 
				
			||||||
	static void dataReplace(QQmlListProperty<QObject>* prop, qsizetype i, QObject* obj);
 | 
						static void dataReplace(QQmlListProperty<QObject>* prop, qsizetype i, QObject* obj);
 | 
				
			||||||
	static void dataRemoveLast(QQmlListProperty<QObject>* prop);
 | 
						static void dataRemoveLast(QQmlListProperty<QObject>* prop);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						PendingRegion* mMask = nullptr;
 | 
				
			||||||
};
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
// qt attempts to resize the window but fails because wayland
 | 
					// qt attempts to resize the window but fails because wayland
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
							
								
								
									
										107
									
								
								src/cpp/region.cpp
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										107
									
								
								src/cpp/region.cpp
									
										
									
									
									
										Normal file
									
								
							| 
						 | 
					@ -0,0 +1,107 @@
 | 
				
			||||||
 | 
					#include "region.hpp"
 | 
				
			||||||
 | 
					#include <cmath>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#include <qobject.h>
 | 
				
			||||||
 | 
					#include <qpoint.h>
 | 
				
			||||||
 | 
					#include <qqmllist.h>
 | 
				
			||||||
 | 
					#include <qquickitem.h>
 | 
				
			||||||
 | 
					#include <qregion.h>
 | 
				
			||||||
 | 
					#include <qtmetamacros.h>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					PendingRegion::PendingRegion(QObject* parent): QObject(parent) {
 | 
				
			||||||
 | 
						QObject::connect(this, &PendingRegion::shapeChanged, this, &PendingRegion::changed);
 | 
				
			||||||
 | 
						QObject::connect(this, &PendingRegion::intersectionChanged, this, &PendingRegion::changed);
 | 
				
			||||||
 | 
						QObject::connect(this, &PendingRegion::itemChanged, this, &PendingRegion::changed);
 | 
				
			||||||
 | 
						QObject::connect(this, &PendingRegion::xChanged, this, &PendingRegion::changed);
 | 
				
			||||||
 | 
						QObject::connect(this, &PendingRegion::yChanged, this, &PendingRegion::changed);
 | 
				
			||||||
 | 
						QObject::connect(this, &PendingRegion::widthChanged, this, &PendingRegion::changed);
 | 
				
			||||||
 | 
						QObject::connect(this, &PendingRegion::heightChanged, this, &PendingRegion::changed);
 | 
				
			||||||
 | 
						QObject::connect(this, &PendingRegion::childrenChanged, this, &PendingRegion::changed);
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					void PendingRegion::setItem(QQuickItem* item) {
 | 
				
			||||||
 | 
						if (this->mItem != nullptr) {
 | 
				
			||||||
 | 
							QObject::disconnect(this->mItem, nullptr, this, nullptr);
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						this->mItem = item;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						QObject::connect(this->mItem, &QQuickItem::xChanged, this, &PendingRegion::itemChanged);
 | 
				
			||||||
 | 
						QObject::connect(this->mItem, &QQuickItem::yChanged, this, &PendingRegion::itemChanged);
 | 
				
			||||||
 | 
						QObject::connect(this->mItem, &QQuickItem::widthChanged, this, &PendingRegion::itemChanged);
 | 
				
			||||||
 | 
						QObject::connect(this->mItem, &QQuickItem::heightChanged, this, &PendingRegion::itemChanged);
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					void PendingRegion::onItemDestroyed() { this->mItem = nullptr; }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					QQmlListProperty<PendingRegion> PendingRegion::regions() {
 | 
				
			||||||
 | 
						return QQmlListProperty<PendingRegion>(
 | 
				
			||||||
 | 
						    this,
 | 
				
			||||||
 | 
						    nullptr,
 | 
				
			||||||
 | 
						    PendingRegion::regionsAppend,
 | 
				
			||||||
 | 
						    nullptr,
 | 
				
			||||||
 | 
						    nullptr,
 | 
				
			||||||
 | 
						    nullptr,
 | 
				
			||||||
 | 
						    nullptr,
 | 
				
			||||||
 | 
						    nullptr
 | 
				
			||||||
 | 
						);
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					bool PendingRegion::empty() const {
 | 
				
			||||||
 | 
						return this->mItem == nullptr && this->mX == 0 && this->mY == 0 && this->mWidth == 0
 | 
				
			||||||
 | 
						    && this->mHeight == 0;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					QRegion PendingRegion::build() const {
 | 
				
			||||||
 | 
						auto type = QRegion::Rectangle;
 | 
				
			||||||
 | 
						switch (this->mShape) {
 | 
				
			||||||
 | 
						case RegionShape::Rect: type = QRegion::Rectangle; break;
 | 
				
			||||||
 | 
						case RegionShape::Ellipse: type = QRegion::Ellipse; break;
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						QRegion region;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if (this->empty()) {
 | 
				
			||||||
 | 
							region = QRegion();
 | 
				
			||||||
 | 
						} else if (this->mItem != nullptr) {
 | 
				
			||||||
 | 
							auto origin = this->mItem->mapToScene(QPointF(0, 0));
 | 
				
			||||||
 | 
							auto extent = this->mItem->mapToScene(QPointF(this->mItem->width(), this->mItem->height()));
 | 
				
			||||||
 | 
							auto size = extent - origin;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							region = QRegion(
 | 
				
			||||||
 | 
							    static_cast<int>(origin.x()),
 | 
				
			||||||
 | 
							    static_cast<int>(origin.y()),
 | 
				
			||||||
 | 
							    static_cast<int>(std::ceil(size.x())),
 | 
				
			||||||
 | 
							    static_cast<int>(std::ceil(size.y())),
 | 
				
			||||||
 | 
							    type
 | 
				
			||||||
 | 
							);
 | 
				
			||||||
 | 
						} else {
 | 
				
			||||||
 | 
							region = QRegion(this->mX, this->mY, this->mWidth, this->mHeight, type);
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						for (const auto& childRegion: this->mRegions) {
 | 
				
			||||||
 | 
							region = childRegion->applyTo(region);
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return region;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					QRegion PendingRegion::applyTo(QRegion& region) const {
 | 
				
			||||||
 | 
						switch (this->mIntersection) {
 | 
				
			||||||
 | 
						case Intersection::Combine: region = region.united(this->build()); break;
 | 
				
			||||||
 | 
						case Intersection::Subtract: region = region.subtracted(this->build()); break;
 | 
				
			||||||
 | 
						case Intersection::Intersect: region = region.intersected(this->build()); break;
 | 
				
			||||||
 | 
						case Intersection::Xor: region = region.xored(this->build()); break;
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return region;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					void PendingRegion::regionsAppend(QQmlListProperty<PendingRegion>* prop, PendingRegion* region) {
 | 
				
			||||||
 | 
						auto* self = static_cast<PendingRegion*>(prop->object); // NOLINT
 | 
				
			||||||
 | 
						region->setParent(self);
 | 
				
			||||||
 | 
						self->mRegions.append(region);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						QObject::connect(region, &PendingRegion::changed, self, &PendingRegion::childrenChanged);
 | 
				
			||||||
 | 
						emit self->childrenChanged();
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
							
								
								
									
										125
									
								
								src/cpp/region.hpp
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										125
									
								
								src/cpp/region.hpp
									
										
									
									
									
										Normal file
									
								
							| 
						 | 
					@ -0,0 +1,125 @@
 | 
				
			||||||
 | 
					#pragma once
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#include <qobject.h>
 | 
				
			||||||
 | 
					#include <qqmlengine.h>
 | 
				
			||||||
 | 
					#include <qqmlintegration.h>
 | 
				
			||||||
 | 
					#include <qqmllist.h>
 | 
				
			||||||
 | 
					#include <qquickitem.h>
 | 
				
			||||||
 | 
					#include <qregion.h>
 | 
				
			||||||
 | 
					#include <qtmetamacros.h>
 | 
				
			||||||
 | 
					#include <qtypes.h>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					/// Shape of a Region.
 | 
				
			||||||
 | 
					namespace RegionShape { // NOLINT
 | 
				
			||||||
 | 
					Q_NAMESPACE;
 | 
				
			||||||
 | 
					QML_ELEMENT;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					enum Enum {
 | 
				
			||||||
 | 
						Rect = 0,
 | 
				
			||||||
 | 
						Ellipse = 1,
 | 
				
			||||||
 | 
					};
 | 
				
			||||||
 | 
					Q_ENUM_NS(Enum);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					} // namespace RegionShape
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					///! Intersection strategy for Regions.
 | 
				
			||||||
 | 
					namespace Intersection { // NOLINT
 | 
				
			||||||
 | 
					Q_NAMESPACE;
 | 
				
			||||||
 | 
					QML_ELEMENT;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					enum Enum {
 | 
				
			||||||
 | 
						/// Combine this region, leaving a union of this and the other region. (opposite of `Subtract`)
 | 
				
			||||||
 | 
						Combine = 0,
 | 
				
			||||||
 | 
						/// Subtract this region, cutting this region out of the other. (opposite of `Combine`)
 | 
				
			||||||
 | 
						Subtract = 1,
 | 
				
			||||||
 | 
						/// Create an intersection of this region and the other, leaving only
 | 
				
			||||||
 | 
						/// the area covered by both. (opposite of `Xor`)
 | 
				
			||||||
 | 
						Intersect = 2,
 | 
				
			||||||
 | 
						/// Create an intersection of this region and the other, leaving only
 | 
				
			||||||
 | 
						/// the area not covered by both. (opposite of `Intersect`)
 | 
				
			||||||
 | 
						Xor = 3,
 | 
				
			||||||
 | 
					};
 | 
				
			||||||
 | 
					Q_ENUM_NS(Enum);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					} // namespace Intersection
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					///! A composable region used as a mask.
 | 
				
			||||||
 | 
					class PendingRegion: public QObject {
 | 
				
			||||||
 | 
						Q_OBJECT;
 | 
				
			||||||
 | 
						/// Defaults to `Rect`.
 | 
				
			||||||
 | 
						Q_PROPERTY(RegionShape::Enum shape MEMBER mShape NOTIFY shapeChanged);
 | 
				
			||||||
 | 
						/// The way this region interacts with its parent region. Defaults to `Combine`.
 | 
				
			||||||
 | 
						Q_PROPERTY(Intersection::Enum intersection MEMBER mIntersection NOTIFY intersectionChanged);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						/// The item that determines the geometry of the region.
 | 
				
			||||||
 | 
						/// `item` overrides `x`, `y`, `width` and `height`.
 | 
				
			||||||
 | 
						Q_PROPERTY(QQuickItem* item MEMBER mItem WRITE setItem NOTIFY itemChanged);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						/// Defaults to 0. Does nothing if `item` is set.
 | 
				
			||||||
 | 
						Q_PROPERTY(qint32 x MEMBER mX NOTIFY xChanged);
 | 
				
			||||||
 | 
						/// Defaults to 0. Does nothing if `item` is set.
 | 
				
			||||||
 | 
						Q_PROPERTY(qint32 y MEMBER mY NOTIFY yChanged);
 | 
				
			||||||
 | 
						/// Defaults to 0. Does nothing if `item` is set.
 | 
				
			||||||
 | 
						Q_PROPERTY(qint32 width MEMBER mWidth NOTIFY widthChanged);
 | 
				
			||||||
 | 
						/// Defaults to 0. Does nothing if `item` is set.
 | 
				
			||||||
 | 
						Q_PROPERTY(qint32 height MEMBER mHeight NOTIFY heightChanged);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						/// Regions to apply on top of this region.
 | 
				
			||||||
 | 
						///
 | 
				
			||||||
 | 
						/// Regions can be nested to create a more complex region.
 | 
				
			||||||
 | 
						/// For example this will create a square region with a cutout in the middle.
 | 
				
			||||||
 | 
						/// ```qml
 | 
				
			||||||
 | 
						/// Region {
 | 
				
			||||||
 | 
						///   width: 100; height: 100;
 | 
				
			||||||
 | 
						///
 | 
				
			||||||
 | 
						///   Region {
 | 
				
			||||||
 | 
						///     x: 50; y: 50;
 | 
				
			||||||
 | 
						///     width: 50; height: 50;
 | 
				
			||||||
 | 
						///     intersection: Intersection.Subtract
 | 
				
			||||||
 | 
						///   }
 | 
				
			||||||
 | 
						/// }
 | 
				
			||||||
 | 
						/// ```
 | 
				
			||||||
 | 
						Q_PROPERTY(QQmlListProperty<PendingRegion> regions READ regions);
 | 
				
			||||||
 | 
						Q_CLASSINFO("DefaultProperty", "regions");
 | 
				
			||||||
 | 
						QML_NAMED_ELEMENT(Region);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					public:
 | 
				
			||||||
 | 
						explicit PendingRegion(QObject* parent = nullptr);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						void setItem(QQuickItem* item);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						QQmlListProperty<PendingRegion> regions();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						[[nodiscard]] bool empty() const;
 | 
				
			||||||
 | 
						[[nodiscard]] QRegion build() const;
 | 
				
			||||||
 | 
						[[nodiscard]] QRegion applyTo(QRegion& region) const;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						RegionShape::Enum mShape = RegionShape::Rect;
 | 
				
			||||||
 | 
						Intersection::Enum mIntersection = Intersection::Combine;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					signals:
 | 
				
			||||||
 | 
						void shapeChanged();
 | 
				
			||||||
 | 
						void intersectionChanged();
 | 
				
			||||||
 | 
						void itemChanged();
 | 
				
			||||||
 | 
						void xChanged();
 | 
				
			||||||
 | 
						void yChanged();
 | 
				
			||||||
 | 
						void widthChanged();
 | 
				
			||||||
 | 
						void heightChanged();
 | 
				
			||||||
 | 
						void childrenChanged();
 | 
				
			||||||
 | 
						void changed();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					private slots:
 | 
				
			||||||
 | 
						void onItemDestroyed();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					private:
 | 
				
			||||||
 | 
						static void regionsAppend(QQmlListProperty<PendingRegion>* prop, PendingRegion* region);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						QQuickItem* mItem = nullptr;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						qint32 mX = 0;
 | 
				
			||||||
 | 
						qint32 mY = 0;
 | 
				
			||||||
 | 
						qint32 mWidth = 0;
 | 
				
			||||||
 | 
						qint32 mHeight = 0;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						QList<PendingRegion*> mRegions;
 | 
				
			||||||
 | 
					};
 | 
				
			||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue