| | |
| | | |
| | | #include <QtGui/QPen> |
| | | #include <QtGui/QPainter> |
| | | #include <QtGui/QScreen> |
| | | |
| | | #include "qwkglobal_p.h" |
| | | |
| | | namespace QWK { |
| | | |
| | | AbstractWindowContext::AbstractWindowContext() { |
| | | } |
| | | AbstractWindowContext::AbstractWindowContext() = default; |
| | | |
| | | AbstractWindowContext::~AbstractWindowContext() = default; |
| | | |
| | | bool AbstractWindowContext::setup(QObject *host, WindowItemDelegate *delegate) { |
| | | if (!host || !delegate) { |
| | | return false; |
| | | void AbstractWindowContext::setup(QObject *host, WindowItemDelegate *delegate) { |
| | | if (m_host || !host || !delegate) { |
| | | return; |
| | | } |
| | | |
| | | auto windowHandle = delegate->hostWindow(host); |
| | | if (!windowHandle) { |
| | | return false; |
| | | } |
| | | |
| | | m_host = host; |
| | | m_delegate.reset(delegate); |
| | | m_windowHandle = windowHandle; |
| | | |
| | | if (!setupHost()) { |
| | | m_host = nullptr; |
| | | m_delegate = nullptr; |
| | | m_windowHandle = nullptr; |
| | | return false; |
| | | m_windowHandle = m_delegate->hostWindow(m_host); |
| | | if (m_windowHandle) { |
| | | winIdChanged(); |
| | | } |
| | | return true; |
| | | } |
| | | |
| | | bool AbstractWindowContext::setWindowAttribute(const QString &key, const QVariant &attribute) { |
| | | auto it = m_windowAttributes.find(key); |
| | | if (it.value() == attribute) |
| | | return true; |
| | | |
| | | auto newVar = attribute; |
| | | auto oldVar = it.value(); |
| | | bool res = false; |
| | | void *args[] = { |
| | | &const_cast<QString &>(key), |
| | | &newVar, |
| | | &oldVar, |
| | | &res, |
| | | }; |
| | | virtual_hook(WindowAttributeChangedHook, args); |
| | | if (res) { |
| | | it.value() = newVar; |
| | | } |
| | | return res; |
| | | } |
| | | |
| | | bool AbstractWindowContext::setHitTestVisible(const QObject *obj, bool visible) { |
| | |
| | | return true; |
| | | } |
| | | |
| | | bool AbstractWindowContext::setHitTestVisible(const QRect &rect, bool visible) { |
| | | Q_ASSERT(rect.isValid()); |
| | | if (!rect.isValid()) { |
| | | return false; |
| | | } |
| | | |
| | | if (visible) { |
| | | m_hitTestVisibleRects.append(rect); |
| | | } else { |
| | | m_hitTestVisibleRects.removeAll(rect); |
| | | } |
| | | hitTestVisibleShapeDirty = true; |
| | | return true; |
| | | } |
| | | |
| | | bool AbstractWindowContext::setSystemButton(WindowAgentBase::SystemButton button, |
| | | QObject *obj) { |
| | | Q_ASSERT(obj); |
| | | Q_ASSERT(button != WindowAgentBase::Unknown); |
| | | if (!obj || (button == WindowAgentBase::Unknown)) { |
| | | if (button == WindowAgentBase::Unknown) { |
| | | return false; |
| | | } |
| | | |
| | |
| | | |
| | | bool AbstractWindowContext::setTitleBar(QObject *item) { |
| | | Q_ASSERT(item); |
| | | if (!item) { |
| | | return false; |
| | | } |
| | | |
| | | if (m_titleBar == item) { |
| | | return false; |
| | | } |
| | |
| | | return true; |
| | | } |
| | | |
| | | void AbstractWindowContext::showSystemMenu(const QPoint &pos) { |
| | | virtual_hook(ShowSystemMenuHook, &const_cast<QPoint &>(pos)); |
| | | #ifdef Q_OS_MAC |
| | | void AbstractWindowContext::setSystemButtonArea(const QRect &rect) { |
| | | m_systemButtonArea = rect; |
| | | virtual_hook(SystemButtonAreaChangedHook, nullptr); |
| | | } |
| | | |
| | | QRegion AbstractWindowContext::hitTestShape() const { |
| | | if (hitTestVisibleShapeDirty) { |
| | | hitTestVisibleShape = {}; |
| | | for (const auto &rect : m_hitTestVisibleRects) { |
| | | hitTestVisibleShape += rect; |
| | | } |
| | | hitTestVisibleShapeDirty = false; |
| | | } |
| | | return hitTestVisibleShape; |
| | | } |
| | | #endif |
| | | |
| | | bool AbstractWindowContext::isInSystemButtons(const QPoint &pos, |
| | | WindowAgentBase::SystemButton *button) const { |
| | |
| | | } |
| | | } |
| | | |
| | | if (!m_hitTestVisibleRects.isEmpty() && hitTestShape().contains(pos)) { |
| | | return false; |
| | | } |
| | | return true; |
| | | } |
| | | |
| | |
| | | return {}; |
| | | } |
| | | |
| | | static constexpr struct { |
| | | const quint32 lilac = MAKE_RGBA_COLOR(210, 233, 189, 226); |
| | | const quint32 pink = MAKE_RGBA_COLOR(177, 205, 190, 240); |
| | | const quint32 tile = MAKE_RGBA_COLOR(193, 195, 211, 203); |
| | | const quint32 azure = MAKE_RGBA_COLOR(25, 90, 190, 255); |
| | | 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 ShowSystemMenuHook: { |
| | | const auto &pos = *reinterpret_cast<const QPoint *>(data); |
| | | // ... |
| | | 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 NeedsDrawBordersHook: { |
| | | auto &result = *reinterpret_cast<bool *>(data); |
| | | result = false; |
| | | break; |
| | | } |
| | | case DrawBordersHook: { |
| | | auto args = reinterpret_cast<void **>(data); |
| | | auto &painter = *reinterpret_cast<QPainter *>(args[0]); |
| | | auto &rect = *reinterpret_cast<const QRect *>(args[1]); |
| | | |
| | | QPen pen; |
| | | pen.setWidth(1); |
| | | |
| | | // Top |
| | | pen.setColor(kSampleColorSet.lilac); |
| | | painter.setPen(pen); |
| | | painter.drawLine(rect.topLeft(), rect.topRight()); |
| | | |
| | | // Right |
| | | pen.setColor(kSampleColorSet.pink); |
| | | painter.setPen(pen); |
| | | painter.drawLine(rect.topRight(), rect.bottomRight()); |
| | | |
| | | // Bottom |
| | | pen.setColor(kSampleColorSet.tile); |
| | | painter.setPen(pen); |
| | | painter.drawLine(rect.bottomLeft(), rect.bottomRight()); |
| | | |
| | | // Right |
| | | pen.setColor(kSampleColorSet.azure); |
| | | painter.setPen(pen); |
| | | painter.drawLine(rect.topLeft(), rect.bottomLeft()); |
| | | 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<QMap<QString, QColor> *>(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<QPoint &>(pos)); |
| | | } |
| | | |
| | | void AbstractWindowContext::notifyWinIdChange() { |
| | | auto oldWindow = m_windowHandle; |
| | | m_windowHandle = m_delegate->hostWindow(m_host); |
| | | if (oldWindow == m_windowHandle) |
| | | return; |
| | | winIdChanged(); |
| | | } |
| | | |
| | | } |