#include "abstractwindowcontext_p.h" #include #include #include #include #include "qwkglobal_p.h" namespace QWK { class WinIdChangeEventFilter : public QObject { public: explicit WinIdChangeEventFilter(QObject *widget, AbstractWindowContext *ctx, QObject *parent = nullptr) : QObject(parent), ctx(ctx) { widget->installEventFilter(this); } protected: bool eventFilter(QObject *obj, QEvent *event) override { Q_UNUSED(obj) if (event->type() == QEvent::WinIdChange) { ctx->notifyWinIdChange(); } return false; } protected: AbstractWindowContext *ctx; }; AbstractWindowContext::AbstractWindowContext() = default; AbstractWindowContext::~AbstractWindowContext() = default; void AbstractWindowContext::setup(QObject *host, WindowItemDelegate *delegate) { if (m_host || !host || !delegate) { return; } m_host = host; m_delegate.reset(delegate); m_winIdChangeEventFilter = std::make_unique(host, this); m_windowHandle = m_delegate->hostWindow(m_host); if (m_windowHandle) { winIdChanged(); } } bool AbstractWindowContext::setWindowAttribute(const QString &key, const QVariant &attribute) { auto it = m_windowAttributes.find(key); if (it == m_windowAttributes.end()) { if (!attribute.isValid()) { return true; } if (m_windowHandle && !windowAttributeChanged(key, attribute, {})) { return false; } m_windowAttributes.insert(key, attribute); return true; } if (it.value() == attribute) return true; if (m_windowHandle && !windowAttributeChanged(key, attribute, it.value())) { return false; } if (attribute.isValid()) { it.value() = attribute; } else { m_windowAttributes.erase(it); } return true; } bool AbstractWindowContext::setHitTestVisible(const QObject *obj, bool visible) { Q_ASSERT(obj); if (!obj) { return false; } if (visible) { m_hitTestVisibleItems.insert(obj); } else { m_hitTestVisibleItems.remove(obj); } return true; } bool AbstractWindowContext::setSystemButton(WindowAgentBase::SystemButton button, QObject *obj) { Q_ASSERT(button != WindowAgentBase::Unknown); if (button == WindowAgentBase::Unknown) { return false; } if (m_systemButtons[button] == obj) { return false; } m_systemButtons[button] = obj; return true; } bool AbstractWindowContext::setTitleBar(QObject *item) { Q_ASSERT(item); if (m_titleBar == item) { return false; } if (m_titleBar) { // Since the title bar is changed, all items inside it should be dereferenced right away for (auto &button : m_systemButtons) { button = nullptr; } m_hitTestVisibleItems.clear(); } m_titleBar = item; return true; } #ifdef Q_OS_MAC void AbstractWindowContext::setSystemButtonArea(const QRect &rect) { m_systemButtonArea = rect; virtual_hook(SystemButtonAreaChangedHook, nullptr); } #endif bool AbstractWindowContext::isInSystemButtons(const QPoint &pos, WindowAgentBase::SystemButton *button) const { *button = WindowAgentBase::Unknown; for (int i = WindowAgentBase::WindowIcon; i <= WindowAgentBase::Close; ++i) { auto currentButton = m_systemButtons[i]; if (!currentButton || !m_delegate->isVisible(currentButton) || !m_delegate->isEnabled(currentButton)) { continue; } if (m_delegate->mapGeometryToScene(currentButton).contains(pos)) { *button = static_cast(i); return true; } } return false; } bool AbstractWindowContext::isInTitleBarDraggableArea(const QPoint &pos) const { if (!m_titleBar) { // There's no title bar at all, the mouse will always be in the client area. return false; } if (!m_delegate->isVisible(m_titleBar) || !m_delegate->isEnabled(m_titleBar)) { // The title bar is hidden or disabled for some reason, treat it as there's // no title bar. return false; } QRect windowRect = {QPoint(0, 0), m_windowHandle->size()}; QRect titleBarRect = m_delegate->mapGeometryToScene(m_titleBar); if (!titleBarRect.intersects(windowRect)) { // The title bar is totally outside the window for some reason, // also treat it as there's no title bar. return false; } if (!titleBarRect.contains(pos)) { return false; } for (int i = WindowAgentBase::WindowIcon; i <= WindowAgentBase::Close; ++i) { auto currentButton = m_systemButtons[i]; if (currentButton && m_delegate->isVisible(currentButton) && m_delegate->isEnabled(currentButton) && m_delegate->mapGeometryToScene(currentButton).contains(pos)) { return false; } } for (auto widget : m_hitTestVisibleItems) { if (widget && m_delegate->isVisible(widget) && m_delegate->isEnabled(widget) && m_delegate->mapGeometryToScene(widget).contains(pos)) { return false; } } return true; } QString AbstractWindowContext::key() const { return {}; } QWK_USED static constexpr const struct { const quint32 activeLight = MAKE_RGBA_COLOR(210, 233, 189, 226); const quint32 activeDark = MAKE_RGBA_COLOR(177, 205, 190, 240); const quint32 inactiveLight = MAKE_RGBA_COLOR(193, 195, 211, 203); const quint32 inactiveDark = MAKE_RGBA_COLOR(240, 240, 250, 255); } kSampleColorSet; void AbstractWindowContext::virtual_hook(int id, void *data) { switch (id) { case CentralizeHook: { if (!m_windowHandle) return; QRect screenGeometry = m_windowHandle->screen()->geometry(); int x = (screenGeometry.width() - m_windowHandle->width()) / 2; int y = (screenGeometry.height() - m_windowHandle->height()) / 2; QPoint pos(x, y); pos += screenGeometry.topLeft(); m_windowHandle->setPosition(pos); return; } case RaiseWindowHook: { Qt::WindowStates state = m_delegate->getWindowState(m_host); if (state & Qt::WindowMinimized) { m_delegate->setWindowState(m_host, state & ~Qt::WindowMinimized); } m_delegate->bringWindowToTop(m_host); return; } case DefaultColorsHook: { auto &map = *static_cast *>(data); map.clear(); map.insert(QStringLiteral("activeLight"), kSampleColorSet.activeLight); map.insert(QStringLiteral("activeDark"), kSampleColorSet.activeDark); map.insert(QStringLiteral("inactiveLight"), kSampleColorSet.inactiveLight); map.insert(QStringLiteral("inactiveDark"), kSampleColorSet.inactiveDark); return; } default: break; } } void AbstractWindowContext::showSystemMenu(const QPoint &pos) { virtual_hook(ShowSystemMenuHook, &const_cast(pos)); } void AbstractWindowContext::notifyWinIdChange() { auto oldWindow = m_windowHandle; m_windowHandle = m_delegate->hostWindow(m_host); if (oldWindow == m_windowHandle) return; winIdChanged(); if (m_windowHandle) { // Refresh window attributes auto attributes = m_windowAttributes; m_windowAttributes.clear(); for (auto it = attributes.begin(); it != attributes.end(); ++it) { if (!windowAttributeChanged(it.key(), it.value(), {})) { continue; } m_windowAttributes.insert(it.key(), it.value()); } } } bool AbstractWindowContext::windowAttributeChanged(const QString &key, const QVariant &attribute, const QVariant &oldAttribute) { return false; } }