2007-11-27 19:40:25 +00:00
|
|
|
/********************************************************************
|
2007-04-29 17:35:43 +00:00
|
|
|
KWin - the KDE window manager
|
|
|
|
This file is part of the KDE project.
|
|
|
|
|
|
|
|
Copyright (C) 2006 Lubos Lunak <l.lunak@kde.org>
|
|
|
|
|
2007-11-27 19:40:25 +00:00
|
|
|
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/>.
|
|
|
|
*********************************************************************/
|
2007-04-29 17:35:43 +00:00
|
|
|
#include "toplevel.h"
|
|
|
|
|
2013-04-04 14:14:12 +00:00
|
|
|
#ifdef KWIN_BUILD_ACTIVITIES
|
|
|
|
#include "activities.h"
|
|
|
|
#endif
|
2007-04-29 17:35:43 +00:00
|
|
|
#include "atoms.h"
|
|
|
|
#include "client.h"
|
2013-01-07 07:07:27 +00:00
|
|
|
#include "client_machine.h"
|
2019-06-22 10:00:56 +00:00
|
|
|
#include "composite.h"
|
2007-04-29 17:35:43 +00:00
|
|
|
#include "effects.h"
|
2013-04-03 10:19:27 +00:00
|
|
|
#include "screens.h"
|
2011-03-27 10:33:07 +00:00
|
|
|
#include "shadow.h"
|
2019-06-22 10:00:56 +00:00
|
|
|
#include "workspace.h"
|
2012-12-21 14:11:31 +00:00
|
|
|
#include "xcbutils.h"
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2015-02-24 09:54:28 +00:00
|
|
|
#include <KWayland/Server/surface_interface.h>
|
|
|
|
|
2013-09-02 11:14:39 +00:00
|
|
|
#include <QDebug>
|
|
|
|
|
2007-04-29 17:35:43 +00:00
|
|
|
namespace KWin
|
|
|
|
{
|
|
|
|
|
2013-05-08 11:39:06 +00:00
|
|
|
Toplevel::Toplevel()
|
2014-04-25 10:06:39 +00:00
|
|
|
: m_visual(XCB_NONE)
|
2015-06-01 14:25:21 +00:00
|
|
|
, bit_depth(24)
|
2011-01-30 14:34:42 +00:00
|
|
|
, info(NULL)
|
2012-10-31 17:01:31 +00:00
|
|
|
, ready_for_painting(true)
|
2012-11-15 06:48:08 +00:00
|
|
|
, m_isDamaged(false)
|
2018-11-18 19:13:55 +00:00
|
|
|
, m_internalId(QUuid::createUuid())
|
2013-09-10 05:09:44 +00:00
|
|
|
, m_client()
|
2011-01-30 14:34:42 +00:00
|
|
|
, damage_handle(None)
|
|
|
|
, is_shape(false)
|
|
|
|
, effect_window(NULL)
|
2013-01-07 07:07:27 +00:00
|
|
|
, m_clientMachine(new ClientMachine(this))
|
2011-01-30 14:34:42 +00:00
|
|
|
, wmClientLeaderWin(0)
|
2012-03-28 18:29:33 +00:00
|
|
|
, m_damageReplyPending(false)
|
2013-03-26 06:45:08 +00:00
|
|
|
, m_screen(0)
|
2014-01-24 11:34:16 +00:00
|
|
|
, m_skipCloseAnimation(false)
|
2011-01-30 14:34:42 +00:00
|
|
|
{
|
2012-08-23 11:42:59 +00:00
|
|
|
connect(this, SIGNAL(damaged(KWin::Toplevel*,QRect)), SIGNAL(needsRepaint()));
|
2013-04-03 10:19:27 +00:00
|
|
|
connect(screens(), SIGNAL(changed()), SLOT(checkScreen()));
|
2013-05-10 20:03:59 +00:00
|
|
|
connect(screens(), SIGNAL(countChanged(int,int)), SLOT(checkScreen()));
|
2013-03-26 06:45:08 +00:00
|
|
|
setupCheckScreenConnection();
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
|
|
|
Toplevel::~Toplevel()
|
2011-01-30 14:34:42 +00:00
|
|
|
{
|
|
|
|
assert(damage_handle == None);
|
2007-04-29 17:35:43 +00:00
|
|
|
delete info;
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
QDebug& operator<<(QDebug& stream, const Toplevel* cl)
|
|
|
|
{
|
|
|
|
if (cl == NULL)
|
2007-04-29 17:35:43 +00:00
|
|
|
return stream << "\'NULL\'";
|
2011-01-30 14:34:42 +00:00
|
|
|
cl->debug(stream);
|
2007-04-29 17:35:43 +00:00
|
|
|
return stream;
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
QDebug& operator<<(QDebug& stream, const ToplevelList& list)
|
|
|
|
{
|
2007-04-29 17:35:43 +00:00
|
|
|
stream << "LIST:(";
|
|
|
|
bool first = true;
|
2011-01-30 14:34:42 +00:00
|
|
|
for (ToplevelList::ConstIterator it = list.begin();
|
|
|
|
it != list.end();
|
|
|
|
++it) {
|
|
|
|
if (!first)
|
2007-04-29 17:35:43 +00:00
|
|
|
stream << ":";
|
|
|
|
first = false;
|
|
|
|
stream << *it;
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
stream << ")";
|
|
|
|
return stream;
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2011-07-13 20:50:31 +00:00
|
|
|
QRect Toplevel::decorationRect() const
|
|
|
|
{
|
2012-03-10 10:34:56 +00:00
|
|
|
return rect();
|
2011-07-13 20:50:31 +00:00
|
|
|
}
|
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
void Toplevel::detectShape(Window id)
|
|
|
|
{
|
2012-01-15 10:03:15 +00:00
|
|
|
const bool wasShape = is_shape;
|
2012-12-21 14:11:31 +00:00
|
|
|
is_shape = Xcb::Extensions::self()->hasShape(id);
|
2012-01-15 10:03:15 +00:00
|
|
|
if (wasShape != is_shape) {
|
|
|
|
emit shapedChanged();
|
|
|
|
}
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
|
|
|
// used only by Deleted::copy()
|
2011-01-30 14:34:42 +00:00
|
|
|
void Toplevel::copyToDeleted(Toplevel* c)
|
|
|
|
{
|
2018-11-18 19:13:55 +00:00
|
|
|
m_internalId = c->internalId();
|
2007-04-29 17:35:43 +00:00
|
|
|
geom = c->geom;
|
2014-04-25 10:06:39 +00:00
|
|
|
m_visual = c->m_visual;
|
2007-04-29 17:35:43 +00:00
|
|
|
bit_depth = c->bit_depth;
|
|
|
|
info = c->info;
|
2013-09-10 05:09:44 +00:00
|
|
|
m_client.reset(c->m_client, false);
|
2007-06-25 08:51:44 +00:00
|
|
|
ready_for_painting = c->ready_for_painting;
|
2007-04-29 17:35:43 +00:00
|
|
|
damage_handle = None;
|
|
|
|
damage_region = c->damage_region;
|
|
|
|
repaints_region = c->repaints_region;
|
2019-02-06 14:09:25 +00:00
|
|
|
layer_repaints_region = c->layer_repaints_region;
|
2007-04-29 17:35:43 +00:00
|
|
|
is_shape = c->is_shape;
|
|
|
|
effect_window = c->effect_window;
|
2011-01-30 14:34:42 +00:00
|
|
|
if (effect_window != NULL)
|
|
|
|
effect_window->setWindow(this);
|
2007-04-29 17:35:43 +00:00
|
|
|
resource_name = c->resourceName();
|
|
|
|
resource_class = c->resourceClass();
|
2013-01-07 07:07:27 +00:00
|
|
|
m_clientMachine = c->m_clientMachine;
|
|
|
|
m_clientMachine->setParent(this);
|
2007-04-29 17:35:43 +00:00
|
|
|
wmClientLeaderWin = c->wmClientLeader();
|
2011-10-22 09:02:49 +00:00
|
|
|
opaque_region = c->opaqueRegion();
|
2013-03-26 06:45:08 +00:00
|
|
|
m_screen = c->m_screen;
|
2014-01-24 11:34:16 +00:00
|
|
|
m_skipCloseAnimation = c->m_skipCloseAnimation;
|
2015-08-18 12:40:26 +00:00
|
|
|
m_internalFBO = c->m_internalFBO;
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
|
|
|
// before being deleted, remove references to everything that's now
|
|
|
|
// owner by Deleted
|
|
|
|
void Toplevel::disownDataPassedToDeleted()
|
2011-01-30 14:34:42 +00:00
|
|
|
{
|
2007-04-29 17:35:43 +00:00
|
|
|
info = NULL;
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2009-06-11 20:12:11 +00:00
|
|
|
QRect Toplevel::visibleRect() const
|
2011-01-30 14:34:42 +00:00
|
|
|
{
|
2012-05-05 14:38:02 +00:00
|
|
|
QRect r = decorationRect();
|
2011-11-25 00:00:54 +00:00
|
|
|
if (hasShadow() && !shadow()->shadowRegion().isEmpty()) {
|
2012-05-05 14:38:02 +00:00
|
|
|
r |= shadow()->shadowRegion().boundingRect();
|
2011-03-27 10:33:07 +00:00
|
|
|
}
|
2012-05-05 14:38:02 +00:00
|
|
|
return r.translated(geometry().topLeft());
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2009-06-11 20:12:11 +00:00
|
|
|
|
2015-01-15 13:33:28 +00:00
|
|
|
Xcb::Property Toplevel::fetchWmClientLeader() const
|
|
|
|
{
|
|
|
|
return Xcb::Property(false, window(), atoms->wm_client_leader, XCB_ATOM_WINDOW, 0, 10000);
|
|
|
|
}
|
|
|
|
|
|
|
|
void Toplevel::readWmClientLeader(Xcb::Property &prop)
|
2011-01-30 14:34:42 +00:00
|
|
|
{
|
2014-04-15 08:09:25 +00:00
|
|
|
wmClientLeaderWin = prop.value<xcb_window_t>(window());
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2015-01-15 13:33:28 +00:00
|
|
|
void Toplevel::getWmClientLeader()
|
|
|
|
{
|
|
|
|
auto prop = fetchWmClientLeader();
|
|
|
|
readWmClientLeader(prop);
|
|
|
|
}
|
|
|
|
|
2019-02-02 18:17:44 +00:00
|
|
|
/**
|
|
|
|
* Returns sessionId for this client,
|
|
|
|
* taken either from its window or from the leader window.
|
2019-07-29 18:58:33 +00:00
|
|
|
*/
|
2012-04-12 19:52:44 +00:00
|
|
|
QByteArray Toplevel::sessionId() const
|
2011-01-30 14:34:42 +00:00
|
|
|
{
|
2014-04-15 08:09:25 +00:00
|
|
|
QByteArray result = Xcb::StringProperty(window(), atoms->sm_client_id);
|
2011-01-30 14:34:42 +00:00
|
|
|
if (result.isEmpty() && wmClientLeaderWin && wmClientLeaderWin != window())
|
2014-04-15 08:09:25 +00:00
|
|
|
result = Xcb::StringProperty(wmClientLeaderWin, atoms->sm_client_id);
|
2007-04-29 17:35:43 +00:00
|
|
|
return result;
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2019-02-02 18:17:44 +00:00
|
|
|
/**
|
|
|
|
* Returns command property for this client,
|
|
|
|
* taken either from its window or from the leader window.
|
2019-07-29 18:58:33 +00:00
|
|
|
*/
|
2016-06-23 17:40:40 +00:00
|
|
|
QByteArray Toplevel::wmCommand()
|
|
|
|
{
|
|
|
|
QByteArray result = Xcb::StringProperty(window(), XCB_ATOM_WM_COMMAND);
|
|
|
|
if (result.isEmpty() && wmClientLeaderWin && wmClientLeaderWin != window())
|
|
|
|
result = Xcb::StringProperty(wmClientLeaderWin, XCB_ATOM_WM_COMMAND);
|
|
|
|
result.replace(0, ' ');
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2007-04-29 17:35:43 +00:00
|
|
|
void Toplevel::getWmClientMachine()
|
2011-01-30 14:34:42 +00:00
|
|
|
{
|
2013-01-07 07:07:27 +00:00
|
|
|
m_clientMachine->resolve(window(), wmClientLeader());
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2019-02-02 18:17:44 +00:00
|
|
|
/**
|
|
|
|
* Returns client machine for this client,
|
|
|
|
* taken either from its window or from the leader window.
|
2019-07-29 18:58:33 +00:00
|
|
|
*/
|
2011-01-30 14:34:42 +00:00
|
|
|
QByteArray Toplevel::wmClientMachine(bool use_localhost) const
|
|
|
|
{
|
2013-01-07 07:07:27 +00:00
|
|
|
if (!m_clientMachine) {
|
|
|
|
// this should never happen
|
|
|
|
return QByteArray();
|
|
|
|
}
|
|
|
|
if (use_localhost && m_clientMachine->isLocal()) {
|
2011-01-30 14:34:42 +00:00
|
|
|
// special name for the local machine (localhost)
|
2013-01-07 07:07:27 +00:00
|
|
|
return ClientMachine::localhost();
|
2007-04-29 17:35:43 +00:00
|
|
|
}
|
2013-01-07 07:07:27 +00:00
|
|
|
return m_clientMachine->hostName();
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2019-02-02 18:17:44 +00:00
|
|
|
/**
|
|
|
|
* Returns client leader window for this client.
|
|
|
|
* Returns the client window itself if no leader window is defined.
|
2019-07-29 18:58:33 +00:00
|
|
|
*/
|
2007-04-29 17:35:43 +00:00
|
|
|
Window Toplevel::wmClientLeader() const
|
2011-01-30 14:34:42 +00:00
|
|
|
{
|
2007-04-29 17:35:43 +00:00
|
|
|
if (wmClientLeaderWin)
|
|
|
|
return wmClientLeaderWin;
|
|
|
|
return window();
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
|
|
|
void Toplevel::getResourceClass()
|
2011-01-30 14:34:42 +00:00
|
|
|
{
|
2015-07-20 07:37:36 +00:00
|
|
|
setResourceClass(QByteArray(info->windowClassName()).toLower(), QByteArray(info->windowClassClass()).toLower());
|
|
|
|
}
|
|
|
|
|
|
|
|
void Toplevel::setResourceClass(const QByteArray &name, const QByteArray &className)
|
|
|
|
{
|
|
|
|
resource_name = name;
|
|
|
|
resource_class = className;
|
2014-04-11 06:06:26 +00:00
|
|
|
emit windowClassChanged();
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2007-09-03 15:00:43 +00:00
|
|
|
double Toplevel::opacity() const
|
2011-01-30 14:34:42 +00:00
|
|
|
{
|
|
|
|
if (info->opacity() == 0xffffffff)
|
2007-04-29 17:35:43 +00:00
|
|
|
return 1.0;
|
|
|
|
return info->opacity() * 1.0 / 0xffffffff;
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
void Toplevel::setOpacity(double new_opacity)
|
|
|
|
{
|
2007-09-03 15:00:43 +00:00
|
|
|
double old_opacity = opacity();
|
2011-01-30 14:34:42 +00:00
|
|
|
new_opacity = qBound(0.0, new_opacity, 1.0);
|
|
|
|
if (old_opacity == new_opacity)
|
2007-04-29 17:35:43 +00:00
|
|
|
return;
|
2011-01-30 14:34:42 +00:00
|
|
|
info->setOpacity(static_cast< unsigned long >(new_opacity * 0xffffffff));
|
|
|
|
if (compositing()) {
|
2007-04-29 17:35:43 +00:00
|
|
|
addRepaintFull();
|
2011-03-06 09:30:23 +00:00
|
|
|
emit opacityChanged(this, old_opacity);
|
2007-04-29 17:35:43 +00:00
|
|
|
}
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2019-06-22 10:00:56 +00:00
|
|
|
bool Toplevel::setupCompositing()
|
|
|
|
{
|
|
|
|
if (!compositing())
|
|
|
|
return false;
|
|
|
|
|
|
|
|
if (damage_handle != XCB_NONE)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
if (kwinApp()->operationMode() == Application::OperationModeX11 && !surface()) {
|
|
|
|
damage_handle = xcb_generate_id(connection());
|
|
|
|
xcb_damage_create(connection(), damage_handle, frameId(), XCB_DAMAGE_REPORT_LEVEL_NON_EMPTY);
|
|
|
|
}
|
|
|
|
|
|
|
|
damage_region = QRegion(0, 0, width(), height());
|
|
|
|
effect_window = new EffectWindowImpl(this);
|
|
|
|
|
|
|
|
Compositor::self()->scene()->addToplevel(this);
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Toplevel::finishCompositing(ReleaseReason releaseReason)
|
|
|
|
{
|
|
|
|
if (kwinApp()->operationMode() == Application::OperationModeX11 && damage_handle == XCB_NONE)
|
|
|
|
return;
|
|
|
|
if (effect_window->window() == this) { // otherwise it's already passed to Deleted, don't free data
|
|
|
|
discardWindowPixmap();
|
|
|
|
delete effect_window;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (damage_handle != XCB_NONE &&
|
|
|
|
releaseReason != ReleaseReason::Destroyed) {
|
|
|
|
xcb_damage_destroy(connection(), damage_handle);
|
|
|
|
}
|
|
|
|
|
|
|
|
damage_handle = XCB_NONE;
|
|
|
|
damage_region = QRegion();
|
|
|
|
repaints_region = QRegion();
|
|
|
|
effect_window = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Toplevel::discardWindowPixmap()
|
|
|
|
{
|
|
|
|
addDamageFull();
|
|
|
|
if (effectWindow() != NULL && effectWindow()->sceneWindow() != NULL)
|
|
|
|
effectWindow()->sceneWindow()->pixmapDiscarded();
|
|
|
|
}
|
|
|
|
|
|
|
|
void Toplevel::damageNotifyEvent()
|
|
|
|
{
|
|
|
|
m_isDamaged = true;
|
|
|
|
|
|
|
|
// Note: The rect is supposed to specify the damage extents,
|
|
|
|
// but we don't know it at this point. No one who connects
|
|
|
|
// to this signal uses the rect however.
|
|
|
|
emit damaged(this, QRect());
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Toplevel::compositing() const
|
|
|
|
{
|
|
|
|
if (!Workspace::self()) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return Workspace::self()->compositing();
|
|
|
|
}
|
|
|
|
|
|
|
|
void Client::damageNotifyEvent()
|
|
|
|
{
|
|
|
|
if (syncRequest.isPending && isResize()) {
|
|
|
|
emit damaged(this, QRect());
|
|
|
|
m_isDamaged = true;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!ready_for_painting) { // avoid "setReadyForPainting()" function calling overhead
|
|
|
|
if (syncRequest.counter == XCB_NONE) { // cannot detect complete redraw, consider done now
|
|
|
|
setReadyForPainting();
|
|
|
|
setupWindowManagementInterface();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Toplevel::damageNotifyEvent();
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Toplevel::resetAndFetchDamage()
|
|
|
|
{
|
|
|
|
if (!m_isDamaged)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
if (damage_handle == XCB_NONE) {
|
|
|
|
m_isDamaged = false;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
xcb_connection_t *conn = connection();
|
|
|
|
|
|
|
|
// Create a new region and copy the damage region to it,
|
|
|
|
// resetting the damaged state.
|
|
|
|
xcb_xfixes_region_t region = xcb_generate_id(conn);
|
|
|
|
xcb_xfixes_create_region(conn, region, 0, 0);
|
|
|
|
xcb_damage_subtract(conn, damage_handle, 0, region);
|
|
|
|
|
|
|
|
// Send a fetch-region request and destroy the region
|
|
|
|
m_regionCookie = xcb_xfixes_fetch_region_unchecked(conn, region);
|
|
|
|
xcb_xfixes_destroy_region(conn, region);
|
|
|
|
|
|
|
|
m_isDamaged = false;
|
|
|
|
m_damageReplyPending = true;
|
|
|
|
|
|
|
|
return m_damageReplyPending;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Toplevel::getDamageRegionReply()
|
|
|
|
{
|
|
|
|
if (!m_damageReplyPending)
|
|
|
|
return;
|
|
|
|
|
|
|
|
m_damageReplyPending = false;
|
|
|
|
|
|
|
|
// Get the fetch-region reply
|
|
|
|
xcb_xfixes_fetch_region_reply_t *reply =
|
|
|
|
xcb_xfixes_fetch_region_reply(connection(), m_regionCookie, 0);
|
|
|
|
|
|
|
|
if (!reply)
|
|
|
|
return;
|
|
|
|
|
|
|
|
// Convert the reply to a QRegion
|
|
|
|
int count = xcb_xfixes_fetch_region_rectangles_length(reply);
|
|
|
|
QRegion region;
|
|
|
|
|
|
|
|
if (count > 1 && count < 16) {
|
|
|
|
xcb_rectangle_t *rects = xcb_xfixes_fetch_region_rectangles(reply);
|
|
|
|
|
|
|
|
QVector<QRect> qrects;
|
|
|
|
qrects.reserve(count);
|
|
|
|
|
|
|
|
for (int i = 0; i < count; i++)
|
|
|
|
qrects << QRect(rects[i].x, rects[i].y, rects[i].width, rects[i].height);
|
|
|
|
|
|
|
|
region.setRects(qrects.constData(), count);
|
|
|
|
} else
|
|
|
|
region += QRect(reply->extents.x, reply->extents.y,
|
|
|
|
reply->extents.width, reply->extents.height);
|
|
|
|
|
|
|
|
damage_region += region;
|
|
|
|
repaints_region += region;
|
|
|
|
|
|
|
|
free(reply);
|
|
|
|
}
|
|
|
|
|
|
|
|
void Toplevel::addDamageFull()
|
|
|
|
{
|
|
|
|
if (!compositing())
|
|
|
|
return;
|
|
|
|
|
|
|
|
damage_region = rect();
|
|
|
|
repaints_region |= rect();
|
|
|
|
|
|
|
|
emit damaged(this, rect());
|
|
|
|
}
|
|
|
|
|
|
|
|
void Toplevel::resetDamage()
|
|
|
|
{
|
|
|
|
damage_region = QRegion();
|
|
|
|
}
|
|
|
|
|
|
|
|
void Toplevel::addRepaint(const QRect& r)
|
|
|
|
{
|
|
|
|
if (!compositing()) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
repaints_region += r;
|
|
|
|
emit needsRepaint();
|
|
|
|
}
|
|
|
|
|
|
|
|
void Toplevel::addRepaint(int x, int y, int w, int h)
|
|
|
|
{
|
|
|
|
QRect r(x, y, w, h);
|
|
|
|
addRepaint(r);
|
|
|
|
}
|
|
|
|
|
|
|
|
void Toplevel::addRepaint(const QRegion& r)
|
|
|
|
{
|
|
|
|
if (!compositing()) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
repaints_region += r;
|
|
|
|
emit needsRepaint();
|
|
|
|
}
|
|
|
|
|
|
|
|
void Toplevel::addLayerRepaint(const QRect& r)
|
|
|
|
{
|
|
|
|
if (!compositing()) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
layer_repaints_region += r;
|
|
|
|
emit needsRepaint();
|
|
|
|
}
|
|
|
|
|
|
|
|
void Toplevel::addLayerRepaint(int x, int y, int w, int h)
|
|
|
|
{
|
|
|
|
QRect r(x, y, w, h);
|
|
|
|
addLayerRepaint(r);
|
|
|
|
}
|
|
|
|
|
|
|
|
void Toplevel::addLayerRepaint(const QRegion& r)
|
|
|
|
{
|
|
|
|
if (!compositing())
|
|
|
|
return;
|
|
|
|
layer_repaints_region += r;
|
|
|
|
emit needsRepaint();
|
|
|
|
}
|
|
|
|
|
|
|
|
void Toplevel::addRepaintFull()
|
|
|
|
{
|
|
|
|
repaints_region = visibleRect().translated(-pos());
|
|
|
|
emit needsRepaint();
|
|
|
|
}
|
|
|
|
|
|
|
|
void Toplevel::resetRepaints()
|
|
|
|
{
|
|
|
|
repaints_region = QRegion();
|
|
|
|
layer_repaints_region = QRegion();
|
|
|
|
}
|
|
|
|
|
|
|
|
void Toplevel::addWorkspaceRepaint(int x, int y, int w, int h)
|
|
|
|
{
|
|
|
|
addWorkspaceRepaint(QRect(x, y, w, h));
|
|
|
|
}
|
|
|
|
|
|
|
|
void Toplevel::addWorkspaceRepaint(const QRect& r2)
|
|
|
|
{
|
|
|
|
if (!compositing())
|
|
|
|
return;
|
|
|
|
Compositor::self()->addRepaint(r2);
|
|
|
|
}
|
|
|
|
|
2012-01-24 16:00:40 +00:00
|
|
|
void Toplevel::setReadyForPainting()
|
|
|
|
{
|
|
|
|
if (!ready_for_painting) {
|
|
|
|
ready_for_painting = true;
|
|
|
|
if (compositing()) {
|
|
|
|
addRepaintFull();
|
|
|
|
emit windowShown(this);
|
2019-01-01 13:49:51 +00:00
|
|
|
if (auto *cl = dynamic_cast<AbstractClient*>(this)) {
|
2012-01-12 06:42:55 +00:00
|
|
|
if (cl->tabGroup() && cl->tabGroup()->current() == cl)
|
|
|
|
cl->tabGroup()->setCurrent(cl, true);
|
|
|
|
}
|
2012-01-24 16:00:40 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2008-05-23 14:41:27 +00:00
|
|
|
void Toplevel::deleteEffectWindow()
|
2011-01-30 14:34:42 +00:00
|
|
|
{
|
2008-05-23 14:41:27 +00:00
|
|
|
delete effect_window;
|
|
|
|
effect_window = NULL;
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2008-05-23 14:41:27 +00:00
|
|
|
|
2013-03-26 06:45:08 +00:00
|
|
|
void Toplevel::checkScreen()
|
2011-01-30 14:34:42 +00:00
|
|
|
{
|
2013-04-03 10:19:27 +00:00
|
|
|
if (screens()->count() == 1) {
|
2013-03-26 06:45:08 +00:00
|
|
|
if (m_screen != 0) {
|
|
|
|
m_screen = 0;
|
|
|
|
emit screenChanged();
|
|
|
|
}
|
2017-11-01 17:54:21 +00:00
|
|
|
} else {
|
|
|
|
const int s = screens()->number(geometry().center());
|
|
|
|
if (s != m_screen) {
|
|
|
|
m_screen = s;
|
|
|
|
emit screenChanged();
|
|
|
|
}
|
2008-06-09 16:09:56 +00:00
|
|
|
}
|
2017-11-01 17:54:21 +00:00
|
|
|
qreal newScale = screens()->scale(m_screen);
|
|
|
|
if (newScale != m_screenScale) {
|
|
|
|
m_screenScale = newScale;
|
|
|
|
emit screenScaleChanged();
|
2013-03-26 06:45:08 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void Toplevel::setupCheckScreenConnection()
|
|
|
|
{
|
|
|
|
connect(this, SIGNAL(geometryShapeChanged(KWin::Toplevel*,QRect)), SLOT(checkScreen()));
|
|
|
|
connect(this, SIGNAL(geometryChanged()), SLOT(checkScreen()));
|
|
|
|
checkScreen();
|
|
|
|
}
|
|
|
|
|
|
|
|
void Toplevel::removeCheckScreenConnection()
|
|
|
|
{
|
|
|
|
disconnect(this, SIGNAL(geometryShapeChanged(KWin::Toplevel*,QRect)), this, SLOT(checkScreen()));
|
|
|
|
disconnect(this, SIGNAL(geometryChanged()), this, SLOT(checkScreen()));
|
|
|
|
}
|
|
|
|
|
|
|
|
int Toplevel::screen() const
|
|
|
|
{
|
|
|
|
return m_screen;
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2008-06-09 16:09:56 +00:00
|
|
|
|
2017-11-01 17:54:21 +00:00
|
|
|
qreal Toplevel::screenScale() const
|
|
|
|
{
|
|
|
|
return m_screenScale;
|
|
|
|
}
|
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
bool Toplevel::isOnScreen(int screen) const
|
|
|
|
{
|
2013-04-03 10:19:27 +00:00
|
|
|
return screens()->geometry(screen).intersects(geometry());
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Toplevel::isOnActiveScreen() const
|
|
|
|
{
|
|
|
|
return isOnScreen(screens()->current());
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2008-06-09 16:09:56 +00:00
|
|
|
|
2011-03-27 10:33:07 +00:00
|
|
|
void Toplevel::getShadow()
|
|
|
|
{
|
2011-07-02 19:39:01 +00:00
|
|
|
QRect dirtyRect; // old & new shadow region
|
2012-03-24 21:42:29 +00:00
|
|
|
const QRect oldVisibleRect = visibleRect();
|
2011-03-27 10:33:07 +00:00
|
|
|
if (hasShadow()) {
|
2011-07-02 19:39:01 +00:00
|
|
|
dirtyRect = shadow()->shadowRegion().boundingRect();
|
2018-12-06 12:54:27 +00:00
|
|
|
if (!effectWindow()->sceneWindow()->shadow()->updateShadow()) {
|
|
|
|
effectWindow()->sceneWindow()->updateShadow(nullptr);
|
|
|
|
}
|
|
|
|
emit shadowChanged();
|
2011-03-27 10:33:07 +00:00
|
|
|
} else {
|
2011-04-03 09:31:33 +00:00
|
|
|
Shadow::createShadow(this);
|
2011-07-02 19:39:01 +00:00
|
|
|
}
|
|
|
|
if (hasShadow())
|
|
|
|
dirtyRect |= shadow()->shadowRegion().boundingRect();
|
2012-03-24 21:42:29 +00:00
|
|
|
if (oldVisibleRect != visibleRect())
|
|
|
|
emit paddingChanged(this, oldVisibleRect);
|
2011-07-02 19:39:01 +00:00
|
|
|
if (dirtyRect.isValid()) {
|
|
|
|
dirtyRect.translate(pos());
|
2012-03-16 17:53:19 +00:00
|
|
|
addLayerRepaint(dirtyRect);
|
2011-03-27 10:33:07 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Toplevel::hasShadow() const
|
|
|
|
{
|
2011-04-03 09:31:33 +00:00
|
|
|
if (effectWindow() && effectWindow()->sceneWindow()) {
|
|
|
|
return effectWindow()->sceneWindow()->shadow() != NULL;
|
|
|
|
}
|
|
|
|
return false;
|
2011-03-27 10:33:07 +00:00
|
|
|
}
|
|
|
|
|
2011-04-03 09:31:33 +00:00
|
|
|
Shadow *Toplevel::shadow()
|
2011-03-27 10:33:07 +00:00
|
|
|
{
|
2011-04-03 09:31:33 +00:00
|
|
|
if (effectWindow() && effectWindow()->sceneWindow()) {
|
|
|
|
return effectWindow()->sceneWindow()->shadow();
|
|
|
|
} else {
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const Shadow *Toplevel::shadow() const
|
|
|
|
{
|
|
|
|
if (effectWindow() && effectWindow()->sceneWindow()) {
|
|
|
|
return effectWindow()->sceneWindow()->shadow();
|
|
|
|
} else {
|
|
|
|
return NULL;
|
|
|
|
}
|
2011-03-27 10:33:07 +00:00
|
|
|
}
|
2008-06-09 16:09:56 +00:00
|
|
|
|
2014-07-24 06:55:57 +00:00
|
|
|
bool Toplevel::wantsShadowToBeRendered() const
|
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2011-10-22 09:02:49 +00:00
|
|
|
void Toplevel::getWmOpaqueRegion()
|
|
|
|
{
|
2015-01-22 14:46:12 +00:00
|
|
|
const auto rects = info->opaqueRegion();
|
2011-10-22 09:02:49 +00:00
|
|
|
QRegion new_opaque_region;
|
2015-01-22 14:46:12 +00:00
|
|
|
for (const auto &r : rects) {
|
|
|
|
new_opaque_region += QRect(r.pos.x, r.pos.y, r.size.width, r.size.height);
|
|
|
|
}
|
2011-10-22 09:02:49 +00:00
|
|
|
|
|
|
|
opaque_region = new_opaque_region;
|
|
|
|
}
|
|
|
|
|
2012-01-15 09:51:09 +00:00
|
|
|
bool Toplevel::isClient() const
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Toplevel::isDeleted() const
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2013-04-04 14:14:12 +00:00
|
|
|
bool Toplevel::isOnCurrentActivity() const
|
|
|
|
{
|
|
|
|
#ifdef KWIN_BUILD_ACTIVITIES
|
2015-07-07 09:48:42 +00:00
|
|
|
if (!Activities::self()) {
|
|
|
|
return true;
|
|
|
|
}
|
2013-04-04 14:14:12 +00:00
|
|
|
return isOnActivity(Activities::self()->current());
|
|
|
|
#else
|
|
|
|
return true;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2013-04-24 08:46:34 +00:00
|
|
|
void Toplevel::elevate(bool elevate)
|
|
|
|
{
|
|
|
|
if (!effectWindow()) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
effectWindow()->elevate(elevate);
|
2013-07-02 10:45:55 +00:00
|
|
|
addWorkspaceRepaint(visibleRect());
|
2013-04-24 08:46:34 +00:00
|
|
|
}
|
|
|
|
|
2013-04-26 09:18:57 +00:00
|
|
|
pid_t Toplevel::pid() const
|
|
|
|
{
|
|
|
|
return info->pid();
|
|
|
|
}
|
|
|
|
|
2013-09-10 06:13:33 +00:00
|
|
|
xcb_window_t Toplevel::frameId() const
|
|
|
|
{
|
|
|
|
return m_client;
|
|
|
|
}
|
|
|
|
|
2015-01-15 14:19:04 +00:00
|
|
|
Xcb::Property Toplevel::fetchSkipCloseAnimation() const
|
|
|
|
{
|
|
|
|
return Xcb::Property(false, window(), atoms->kde_skip_close_animation, XCB_ATOM_CARDINAL, 0, 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
void Toplevel::readSkipCloseAnimation(Xcb::Property &property)
|
2014-01-24 11:34:16 +00:00
|
|
|
{
|
2014-04-15 08:09:25 +00:00
|
|
|
setSkipCloseAnimation(property.toBool());
|
2014-01-24 11:34:16 +00:00
|
|
|
}
|
|
|
|
|
2015-01-15 14:19:04 +00:00
|
|
|
void Toplevel::getSkipCloseAnimation()
|
|
|
|
{
|
|
|
|
Xcb::Property property = fetchSkipCloseAnimation();
|
|
|
|
readSkipCloseAnimation(property);
|
|
|
|
}
|
|
|
|
|
2014-01-24 11:34:16 +00:00
|
|
|
bool Toplevel::skipsCloseAnimation() const
|
|
|
|
{
|
|
|
|
return m_skipCloseAnimation;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Toplevel::setSkipCloseAnimation(bool set)
|
|
|
|
{
|
|
|
|
if (set == m_skipCloseAnimation) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
m_skipCloseAnimation = set;
|
|
|
|
emit skipCloseAnimationChanged();
|
|
|
|
}
|
|
|
|
|
2015-02-24 09:54:28 +00:00
|
|
|
void Toplevel::setSurface(KWayland::Server::SurfaceInterface *surface)
|
|
|
|
{
|
|
|
|
if (m_surface == surface) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
using namespace KWayland::Server;
|
|
|
|
if (m_surface) {
|
|
|
|
disconnect(m_surface, &SurfaceInterface::damaged, this, &Toplevel::addDamage);
|
2016-08-26 06:32:03 +00:00
|
|
|
disconnect(m_surface, &SurfaceInterface::sizeChanged, this, &Toplevel::discardWindowPixmap);
|
2015-02-24 09:54:28 +00:00
|
|
|
}
|
|
|
|
m_surface = surface;
|
|
|
|
connect(m_surface, &SurfaceInterface::damaged, this, &Toplevel::addDamage);
|
2016-08-26 06:32:03 +00:00
|
|
|
connect(m_surface, &SurfaceInterface::sizeChanged, this, &Toplevel::discardWindowPixmap);
|
2016-03-21 15:24:59 +00:00
|
|
|
connect(m_surface, &SurfaceInterface::subSurfaceTreeChanged, this,
|
|
|
|
[this] {
|
|
|
|
// TODO improve to only update actual visual area
|
|
|
|
if (ready_for_painting) {
|
|
|
|
addDamageFull();
|
|
|
|
m_isDamaged = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
);
|
2015-06-01 14:51:27 +00:00
|
|
|
connect(m_surface, &SurfaceInterface::destroyed, this,
|
|
|
|
[this] {
|
|
|
|
m_surface = nullptr;
|
|
|
|
}
|
|
|
|
);
|
2016-06-26 13:57:38 +00:00
|
|
|
emit surfaceChanged();
|
2015-02-24 09:54:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void Toplevel::addDamage(const QRegion &damage)
|
|
|
|
{
|
|
|
|
m_isDamaged = true;
|
|
|
|
damage_region += damage;
|
2019-07-09 20:08:47 +00:00
|
|
|
for (const QRect &r : damage) {
|
2015-02-24 09:54:28 +00:00
|
|
|
emit damaged(this, r);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-03-04 07:53:51 +00:00
|
|
|
QByteArray Toplevel::windowRole() const
|
|
|
|
{
|
|
|
|
return QByteArray(info->windowRole());
|
|
|
|
}
|
|
|
|
|
2015-06-01 14:25:21 +00:00
|
|
|
void Toplevel::setDepth(int depth)
|
|
|
|
{
|
|
|
|
if (bit_depth == depth) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
const bool oldAlpha = hasAlpha();
|
|
|
|
bit_depth = depth;
|
|
|
|
if (oldAlpha != hasAlpha()) {
|
|
|
|
emit hasAlphaChanged();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-06-18 21:19:13 +00:00
|
|
|
QRegion Toplevel::inputShape() const
|
|
|
|
{
|
|
|
|
if (m_surface) {
|
|
|
|
return m_surface->input();
|
2015-08-10 14:31:45 +00:00
|
|
|
} else {
|
|
|
|
// TODO: maybe also for X11?
|
|
|
|
return QRegion();
|
|
|
|
}
|
2015-06-18 21:19:13 +00:00
|
|
|
}
|
|
|
|
|
2015-08-18 12:40:26 +00:00
|
|
|
void Toplevel::setInternalFramebufferObject(const QSharedPointer<QOpenGLFramebufferObject> &fbo)
|
|
|
|
{
|
|
|
|
if (m_internalFBO != fbo) {
|
|
|
|
discardWindowPixmap();
|
|
|
|
m_internalFBO = fbo;
|
|
|
|
}
|
|
|
|
setDepth(32);
|
|
|
|
}
|
|
|
|
|
2015-12-07 13:25:08 +00:00
|
|
|
QMatrix4x4 Toplevel::inputTransformation() const
|
|
|
|
{
|
|
|
|
QMatrix4x4 m;
|
|
|
|
m.translate(-x(), -y());
|
|
|
|
return m;
|
|
|
|
}
|
|
|
|
|
2016-05-02 15:47:39 +00:00
|
|
|
quint32 Toplevel::windowId() const
|
|
|
|
{
|
|
|
|
return window();
|
|
|
|
}
|
|
|
|
|
2016-09-15 09:44:32 +00:00
|
|
|
QRect Toplevel::inputGeometry() const
|
|
|
|
{
|
|
|
|
return geometry();
|
|
|
|
}
|
|
|
|
|
2018-12-23 07:56:15 +00:00
|
|
|
bool Toplevel::isLocalhost() const
|
|
|
|
{
|
|
|
|
if (!m_clientMachine) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return m_clientMachine->isLocal();
|
|
|
|
}
|
|
|
|
|
2007-04-29 17:35:43 +00:00
|
|
|
} // namespace
|
|
|
|
|