forked from quickshell/quickshell
ProxyWindowBase no longer takes ownership of the mask. It is no longer undefined behavior to set the mask to null.
221 lines
6.4 KiB
C++
221 lines
6.4 KiB
C++
#include "proxywindow.hpp"
|
|
|
|
#include <qobject.h>
|
|
#include <qqmllist.h>
|
|
#include <qquickitem.h>
|
|
#include <qquickwindow.h>
|
|
#include <qregion.h>
|
|
#include <qtmetamacros.h>
|
|
#include <qtypes.h>
|
|
#include <qwindow.h>
|
|
|
|
#include "qmlscreen.hpp"
|
|
#include "region.hpp"
|
|
#include "reload.hpp"
|
|
|
|
ProxyWindowBase::ProxyWindowBase(QObject* parent): Reloadable(parent) {
|
|
this->mContentItem = new QQuickItem(); // NOLINT
|
|
this->mContentItem->setParent(this);
|
|
|
|
QObject::connect(this, &ProxyWindowBase::widthChanged, this, &ProxyWindowBase::onWidthChanged);
|
|
QObject::connect(this, &ProxyWindowBase::heightChanged, this, &ProxyWindowBase::onHeightChanged);
|
|
}
|
|
|
|
ProxyWindowBase::~ProxyWindowBase() {
|
|
if (this->window != nullptr) {
|
|
this->window->deleteLater();
|
|
}
|
|
}
|
|
|
|
void ProxyWindowBase::onReload(QObject* oldInstance) {
|
|
this->window = this->createWindow(oldInstance);
|
|
this->setupWindow();
|
|
|
|
Reloadable::reloadRecursive(this->mContentItem, oldInstance);
|
|
|
|
this->mContentItem->setParentItem(this->window->contentItem());
|
|
|
|
this->mContentItem->setWidth(this->width());
|
|
this->mContentItem->setHeight(this->height());
|
|
|
|
// without this the dangling screen pointer wont be updated to a real screen
|
|
emit this->screenChanged();
|
|
|
|
emit this->windowConnected();
|
|
this->window->setVisible(this->mVisible);
|
|
}
|
|
|
|
QQuickWindow* ProxyWindowBase::createWindow(QObject* oldInstance) {
|
|
auto* old = qobject_cast<ProxyWindowBase*>(oldInstance);
|
|
|
|
if (old == nullptr || old->window == nullptr) {
|
|
return new QQuickWindow();
|
|
} else {
|
|
return old->disownWindow();
|
|
}
|
|
}
|
|
|
|
void ProxyWindowBase::setupWindow() {
|
|
// clang-format off
|
|
QObject::connect(this->window, &QWindow::visibilityChanged, this, &ProxyWindowBase::visibleChanged);
|
|
QObject::connect(this->window, &QWindow::widthChanged, this, &ProxyWindowBase::widthChanged);
|
|
QObject::connect(this->window, &QWindow::heightChanged, this, &ProxyWindowBase::heightChanged);
|
|
QObject::connect(this->window, &QWindow::screenChanged, this, &ProxyWindowBase::screenChanged);
|
|
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
|
|
|
|
this->window->setScreen(this->mScreen);
|
|
this->setWidth(this->mWidth);
|
|
this->setHeight(this->mHeight);
|
|
this->setColor(this->mColor);
|
|
this->updateMask();
|
|
}
|
|
|
|
QQuickWindow* ProxyWindowBase::disownWindow() {
|
|
QObject::disconnect(this->window, nullptr, this, nullptr);
|
|
|
|
this->mContentItem->setParentItem(nullptr);
|
|
|
|
auto* window = this->window;
|
|
this->window = nullptr;
|
|
return window;
|
|
}
|
|
|
|
QQuickWindow* ProxyWindowBase::backingWindow() const { return this->window; }
|
|
QQuickItem* ProxyWindowBase::contentItem() const { return this->mContentItem; }
|
|
|
|
bool ProxyWindowBase::isVisible() const {
|
|
if (this->window == nullptr) return this->mVisible;
|
|
else return this->window->isVisible();
|
|
}
|
|
|
|
void ProxyWindowBase::setVisible(bool visible) {
|
|
if (this->window == nullptr) {
|
|
this->mVisible = visible;
|
|
emit this->visibleChanged();
|
|
} else this->window->setVisible(visible);
|
|
}
|
|
|
|
qint32 ProxyWindowBase::width() const {
|
|
if (this->window == nullptr) return this->mWidth;
|
|
else return this->window->width();
|
|
}
|
|
|
|
void ProxyWindowBase::setWidth(qint32 width) {
|
|
if (this->window == nullptr) {
|
|
this->mWidth = width;
|
|
emit this->widthChanged();
|
|
} else this->window->setWidth(width);
|
|
}
|
|
|
|
qint32 ProxyWindowBase::height() const {
|
|
if (this->window == nullptr) return this->mHeight;
|
|
else return this->window->height();
|
|
}
|
|
|
|
void ProxyWindowBase::setHeight(qint32 height) {
|
|
if (this->window == nullptr) {
|
|
this->mHeight = height;
|
|
emit this->heightChanged();
|
|
} else this->window->setHeight(height);
|
|
}
|
|
|
|
void ProxyWindowBase::setScreen(QuickshellScreenInfo* screen) {
|
|
if (this->mScreen != nullptr) {
|
|
QObject::disconnect(this->mScreen, nullptr, this, nullptr);
|
|
}
|
|
|
|
auto* qscreen = screen == nullptr ? nullptr : screen->screen;
|
|
if (qscreen != nullptr) {
|
|
QObject::connect(qscreen, &QObject::destroyed, this, &ProxyWindowBase::onScreenDestroyed);
|
|
}
|
|
|
|
if (this->window == nullptr) {
|
|
this->mScreen = qscreen;
|
|
emit this->screenChanged();
|
|
} else this->window->setScreen(qscreen);
|
|
}
|
|
|
|
void ProxyWindowBase::onScreenDestroyed() { this->mScreen = nullptr; }
|
|
|
|
QuickshellScreenInfo* ProxyWindowBase::screen() const {
|
|
QScreen* qscreen = nullptr;
|
|
|
|
if (this->window == nullptr) {
|
|
if (this->mScreen != nullptr) qscreen = this->mScreen;
|
|
} else {
|
|
qscreen = this->window->screen();
|
|
}
|
|
|
|
return new QuickshellScreenInfo(
|
|
const_cast<ProxyWindowBase*>(this), // NOLINT
|
|
qscreen
|
|
);
|
|
}
|
|
|
|
QColor ProxyWindowBase::color() const {
|
|
if (this->window == nullptr) return this->mColor;
|
|
else return this->window->color();
|
|
}
|
|
|
|
void ProxyWindowBase::setColor(QColor color) {
|
|
if (this->window == nullptr) {
|
|
this->mColor = color;
|
|
emit this->colorChanged();
|
|
} else this->window->setColor(color);
|
|
}
|
|
|
|
PendingRegion* ProxyWindowBase::mask() const { return this->mMask; }
|
|
|
|
void ProxyWindowBase::setMask(PendingRegion* mask) {
|
|
if (mask == this->mMask) return;
|
|
|
|
if (this->mMask != nullptr) {
|
|
QObject::disconnect(this->mMask, nullptr, this, nullptr);
|
|
}
|
|
|
|
this->mMask = mask;
|
|
|
|
if (mask != nullptr) {
|
|
mask->setParent(this);
|
|
QObject::connect(mask, &QObject::destroyed, this, &ProxyWindowBase::onMaskDestroyed);
|
|
QObject::connect(mask, &PendingRegion::changed, this, &ProxyWindowBase::maskChanged);
|
|
}
|
|
|
|
emit this->maskChanged();
|
|
}
|
|
|
|
void ProxyWindowBase::onMaskChanged() {
|
|
if (this->window != nullptr) this->updateMask();
|
|
}
|
|
|
|
void ProxyWindowBase::onMaskDestroyed() {
|
|
this->mMask = nullptr;
|
|
emit this->maskChanged();
|
|
}
|
|
|
|
void ProxyWindowBase::updateMask() {
|
|
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);
|
|
}
|
|
|
|
QQmlListProperty<QObject> ProxyWindowBase::data() {
|
|
return this->mContentItem->property("data").value<QQmlListProperty<QObject>>();
|
|
}
|
|
|
|
void ProxyWindowBase::onWidthChanged() { this->mContentItem->setWidth(this->width()); }
|
|
void ProxyWindowBase::onHeightChanged() { this->mContentItem->setHeight(this->height()); }
|