2017-08-07 15:54:56 +00:00
|
|
|
/********************************************************************
|
|
|
|
KWin - the KDE window manager
|
|
|
|
This file is part of the KDE project.
|
|
|
|
|
|
|
|
Copyright (C) 2011 Arthur Arlt <a.arlt@stud.uni-heidelberg.de>
|
|
|
|
|
|
|
|
This program is free software; you can redistribute it and/or modify
|
|
|
|
it under the terms of the GNU General Public License as published by
|
|
|
|
the Free Software Foundation; either version 2 of the License, or
|
|
|
|
(at your option) any later version.
|
|
|
|
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
GNU General Public License for more details.
|
|
|
|
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
|
|
along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*********************************************************************/
|
|
|
|
|
|
|
|
#include "overlaywindow_x11.h"
|
|
|
|
|
|
|
|
#include "kwinglobals.h"
|
2017-08-08 05:10:55 +00:00
|
|
|
#include "composite.h"
|
2017-08-07 15:54:56 +00:00
|
|
|
#include "screens.h"
|
|
|
|
#include "utils.h"
|
|
|
|
#include "xcbutils.h"
|
|
|
|
|
|
|
|
#include <QVector>
|
|
|
|
|
|
|
|
#include <xcb/composite.h>
|
|
|
|
#include <xcb/shape.h>
|
|
|
|
#if XCB_COMPOSITE_MAJOR_VERSION > 0 || XCB_COMPOSITE_MINOR_VERSION >= 3
|
|
|
|
#define KWIN_HAVE_XCOMPOSITE_OVERLAY
|
|
|
|
#endif
|
|
|
|
|
|
|
|
namespace KWin {
|
|
|
|
OverlayWindowX11::OverlayWindowX11()
|
|
|
|
: OverlayWindow()
|
2017-08-08 05:10:55 +00:00
|
|
|
, X11EventFilter(QVector<int>{XCB_EXPOSE, XCB_VISIBILITY_NOTIFY})
|
2017-08-07 15:54:56 +00:00
|
|
|
, m_visible(true)
|
|
|
|
, m_shown(false)
|
|
|
|
, m_window(XCB_WINDOW_NONE)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
OverlayWindowX11::~OverlayWindowX11()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
bool OverlayWindowX11::create()
|
|
|
|
{
|
2019-08-31 14:28:37 +00:00
|
|
|
Q_ASSERT(m_window == XCB_WINDOW_NONE);
|
2017-08-07 15:54:56 +00:00
|
|
|
if (!Xcb::Extensions::self()->isCompositeOverlayAvailable())
|
|
|
|
return false;
|
|
|
|
if (!Xcb::Extensions::self()->isShapeInputAvailable()) // needed in setupOverlay()
|
|
|
|
return false;
|
|
|
|
#ifdef KWIN_HAVE_XCOMPOSITE_OVERLAY
|
|
|
|
Xcb::OverlayWindow overlay(rootWindow());
|
|
|
|
if (overlay.isNull()) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
m_window = overlay->overlay_win;
|
|
|
|
if (m_window == XCB_WINDOW_NONE)
|
|
|
|
return false;
|
|
|
|
resize(screens()->size());
|
|
|
|
return true;
|
|
|
|
#else
|
|
|
|
return false;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
void OverlayWindowX11::setup(xcb_window_t window)
|
|
|
|
{
|
2019-08-31 14:28:37 +00:00
|
|
|
Q_ASSERT(m_window != XCB_WINDOW_NONE);
|
|
|
|
Q_ASSERT(Xcb::Extensions::self()->isShapeInputAvailable());
|
2017-08-07 15:54:56 +00:00
|
|
|
setNoneBackgroundPixmap(m_window);
|
|
|
|
m_shape = QRegion();
|
|
|
|
const QSize &s = screens()->size();
|
|
|
|
setShape(QRect(0, 0, s.width(), s.height()));
|
|
|
|
if (window != XCB_WINDOW_NONE) {
|
|
|
|
setNoneBackgroundPixmap(window);
|
|
|
|
setupInputShape(window);
|
|
|
|
}
|
|
|
|
const uint32_t eventMask = XCB_EVENT_MASK_VISIBILITY_CHANGE;
|
|
|
|
xcb_change_window_attributes(connection(), m_window, XCB_CW_EVENT_MASK, &eventMask);
|
|
|
|
}
|
|
|
|
|
|
|
|
void OverlayWindowX11::setupInputShape(xcb_window_t window)
|
|
|
|
{
|
|
|
|
xcb_shape_rectangles(connection(), XCB_SHAPE_SO_SET, XCB_SHAPE_SK_INPUT, XCB_CLIP_ORDERING_UNSORTED, window, 0, 0, 0, NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
void OverlayWindowX11::setNoneBackgroundPixmap(xcb_window_t window)
|
|
|
|
{
|
|
|
|
const uint32_t mask = XCB_BACK_PIXMAP_NONE;
|
|
|
|
xcb_change_window_attributes(connection(), window, XCB_CW_BACK_PIXMAP, &mask);
|
|
|
|
}
|
|
|
|
|
|
|
|
void OverlayWindowX11::show()
|
|
|
|
{
|
2019-08-31 14:28:37 +00:00
|
|
|
Q_ASSERT(m_window != XCB_WINDOW_NONE);
|
2017-08-07 15:54:56 +00:00
|
|
|
if (m_shown)
|
|
|
|
return;
|
|
|
|
xcb_map_subwindows(connection(), m_window);
|
|
|
|
xcb_map_window(connection(), m_window);
|
|
|
|
m_shown = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void OverlayWindowX11::hide()
|
|
|
|
{
|
2019-08-31 14:28:37 +00:00
|
|
|
Q_ASSERT(m_window != XCB_WINDOW_NONE);
|
2017-08-07 15:54:56 +00:00
|
|
|
xcb_unmap_window(connection(), m_window);
|
|
|
|
m_shown = false;
|
|
|
|
const QSize &s = screens()->size();
|
|
|
|
setShape(QRect(0, 0, s.width(), s.height()));
|
|
|
|
}
|
|
|
|
|
|
|
|
void OverlayWindowX11::setShape(const QRegion& reg)
|
|
|
|
{
|
|
|
|
// Avoid setting the same shape again, it causes flicker (apparently it is not a no-op
|
|
|
|
// and triggers something).
|
|
|
|
if (reg == m_shape)
|
|
|
|
return;
|
2019-07-09 20:08:47 +00:00
|
|
|
const QVector<xcb_rectangle_t> xrects = Xcb::regionToRects(reg);
|
2017-08-07 15:54:56 +00:00
|
|
|
xcb_shape_rectangles(connection(), XCB_SHAPE_SO_SET, XCB_SHAPE_SK_BOUNDING, XCB_CLIP_ORDERING_UNSORTED,
|
2019-07-09 20:08:47 +00:00
|
|
|
m_window, 0, 0, xrects.count(), xrects.data());
|
2017-08-07 15:54:56 +00:00
|
|
|
setupInputShape(m_window);
|
|
|
|
m_shape = reg;
|
|
|
|
}
|
|
|
|
|
|
|
|
void OverlayWindowX11::resize(const QSize &size)
|
|
|
|
{
|
2019-08-31 14:28:37 +00:00
|
|
|
Q_ASSERT(m_window != XCB_WINDOW_NONE);
|
2017-08-07 15:54:56 +00:00
|
|
|
const uint32_t geometry[2] = {
|
|
|
|
static_cast<uint32_t>(size.width()),
|
|
|
|
static_cast<uint32_t>(size.height())
|
|
|
|
};
|
|
|
|
xcb_configure_window(connection(), m_window, XCB_CONFIG_WINDOW_WIDTH | XCB_CONFIG_WINDOW_HEIGHT, geometry);
|
|
|
|
setShape(QRegion(0, 0, size.width(), size.height()));
|
|
|
|
}
|
|
|
|
|
|
|
|
bool OverlayWindowX11::isVisible() const
|
|
|
|
{
|
|
|
|
return m_visible;
|
|
|
|
}
|
|
|
|
|
|
|
|
void OverlayWindowX11::setVisibility(bool visible)
|
|
|
|
{
|
|
|
|
m_visible = visible;
|
|
|
|
}
|
|
|
|
|
|
|
|
void OverlayWindowX11::destroy()
|
|
|
|
{
|
|
|
|
if (m_window == XCB_WINDOW_NONE)
|
|
|
|
return;
|
|
|
|
// reset the overlay shape
|
|
|
|
const QSize &s = screens()->size();
|
|
|
|
xcb_rectangle_t rec = { 0, 0, static_cast<uint16_t>(s.width()), static_cast<uint16_t>(s.height()) };
|
|
|
|
xcb_shape_rectangles(connection(), XCB_SHAPE_SO_SET, XCB_SHAPE_SK_BOUNDING, XCB_CLIP_ORDERING_UNSORTED, m_window, 0, 0, 1, &rec);
|
|
|
|
xcb_shape_rectangles(connection(), XCB_SHAPE_SO_SET, XCB_SHAPE_SK_INPUT, XCB_CLIP_ORDERING_UNSORTED, m_window, 0, 0, 1, &rec);
|
|
|
|
#ifdef KWIN_HAVE_XCOMPOSITE_OVERLAY
|
|
|
|
xcb_composite_release_overlay_window(connection(), m_window);
|
|
|
|
#endif
|
|
|
|
m_window = XCB_WINDOW_NONE;
|
|
|
|
m_shown = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
xcb_window_t OverlayWindowX11::window() const
|
|
|
|
{
|
|
|
|
return m_window;
|
|
|
|
}
|
|
|
|
|
2017-08-08 05:10:55 +00:00
|
|
|
bool OverlayWindowX11::event(xcb_generic_event_t *event)
|
|
|
|
{
|
|
|
|
const uint8_t eventType = event->response_type & ~0x80;
|
|
|
|
if (eventType == XCB_EXPOSE) {
|
|
|
|
const auto *expose = reinterpret_cast<xcb_expose_event_t*>(event);
|
|
|
|
if (expose->window == rootWindow() // root window needs repainting
|
|
|
|
|| (m_window != XCB_WINDOW_NONE && expose->window == m_window)) { // overlay needs repainting
|
|
|
|
Compositor::self()->addRepaint(expose->x, expose->y, expose->width, expose->height);
|
|
|
|
}
|
|
|
|
} else if (eventType == XCB_VISIBILITY_NOTIFY) {
|
|
|
|
const auto *visibility = reinterpret_cast<xcb_visibility_notify_event_t*>(event);
|
|
|
|
if (m_window != XCB_WINDOW_NONE && visibility->window == m_window) {
|
|
|
|
bool was_visible = isVisible();
|
|
|
|
setVisibility((visibility->state != XCB_VISIBILITY_FULLY_OBSCURED));
|
|
|
|
auto compositor = Compositor::self();
|
|
|
|
if (!was_visible && m_visible) {
|
|
|
|
// hack for #154825
|
|
|
|
compositor->addRepaintFull();
|
|
|
|
QTimer::singleShot(2000, compositor, &Compositor::addRepaintFull);
|
|
|
|
}
|
|
|
|
compositor->scheduleRepaint();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2017-08-07 15:54:56 +00:00
|
|
|
} // namespace KWin
|
|
|
|
|