4eadc9daef
This is a very interesting auto test as Screens uses both Workspace and Client. Thus it operates in the "impossible to mock" area. The solution is to provide mock includes in autotests and ensure that when building the auto-test the mock header includes will be picked first. There is now a mock class for Workspace and Client providing just the API pieces used inside Screens. As Screens is abstract and we cannot properly interact with QDesktopWidget there is also a MockScreens class inheriting from Screens and mocking the required functionality (by just operating on a list of QRects). The auto-test itself is only performing checks on the abstract class. The mock class is indirectly tested by Screens calling into the virtual methods. The test case is not yet complete, but looking quite good already.
218 lines
5.1 KiB
C++
218 lines
5.1 KiB
C++
/********************************************************************
|
|
KWin - the KDE window manager
|
|
This file is part of the KDE project.
|
|
|
|
Copyright (C) 2013 Martin Gräßlin <mgraesslin@kde.org>
|
|
|
|
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 "screens.h"
|
|
#include <client.h>
|
|
#include "cursor.h"
|
|
#include "settings.h"
|
|
#include <workspace.h>
|
|
#include <config-kwin.h>
|
|
#if HAVE_WAYLAND
|
|
#include "screens_wayland.h"
|
|
#endif
|
|
#ifdef KWIN_UNIT_TEST
|
|
#include <mock_screens.h>
|
|
#endif
|
|
|
|
#include <QApplication>
|
|
#include <QDesktopWidget>
|
|
#include <QTimer>
|
|
|
|
namespace KWin
|
|
{
|
|
|
|
Screens *Screens::s_self = nullptr;
|
|
Screens *Screens::create(QObject *parent)
|
|
{
|
|
Q_ASSERT(!s_self);
|
|
#ifdef KWIN_UNIT_TEST
|
|
s_self = new MockScreens(parent);
|
|
#else
|
|
#if HAVE_WAYLAND
|
|
if (kwinApp()->operationMode() == Application::OperationModeWaylandAndX11) {
|
|
s_self = new WaylandScreens(parent);
|
|
}
|
|
#endif
|
|
if (kwinApp()->operationMode() == Application::OperationModeX11) {
|
|
s_self = new DesktopWidgetScreens(parent);
|
|
}
|
|
#endif
|
|
s_self->init();
|
|
return s_self;
|
|
}
|
|
|
|
Screens::Screens(QObject *parent)
|
|
: QObject(parent)
|
|
, m_count(0)
|
|
, m_current(0)
|
|
, m_currentFollowsMouse(false)
|
|
, m_changedTimer(new QTimer(this))
|
|
{
|
|
}
|
|
|
|
Screens::~Screens()
|
|
{
|
|
s_self = NULL;
|
|
}
|
|
|
|
void Screens::init()
|
|
{
|
|
m_changedTimer->setSingleShot(true);
|
|
m_changedTimer->setInterval(100);
|
|
connect(m_changedTimer, SIGNAL(timeout()), SLOT(updateCount()));
|
|
connect(m_changedTimer, SIGNAL(timeout()), SIGNAL(changed()));
|
|
connect(this, &Screens::countChanged, this, &Screens::changed, Qt::QueuedConnection);
|
|
connect(this, &Screens::changed, this, &Screens::updateSize);
|
|
connect(this, &Screens::sizeChanged, this, &Screens::geometryChanged);
|
|
|
|
Settings settings;
|
|
settings.setDefaults();
|
|
m_currentFollowsMouse = settings.activeMouseScreen();
|
|
}
|
|
|
|
void Screens::reconfigure()
|
|
{
|
|
if (!m_config) {
|
|
return;
|
|
}
|
|
Settings settings(m_config);
|
|
settings.read();
|
|
setCurrentFollowsMouse(settings.activeMouseScreen());
|
|
}
|
|
|
|
void Screens::updateSize()
|
|
{
|
|
QRect bounding;
|
|
for (int i = 0; i < count(); ++i) {
|
|
bounding = bounding.united(geometry(i));
|
|
}
|
|
if (m_boundingSize != bounding.size()) {
|
|
m_boundingSize = bounding.size();
|
|
emit sizeChanged();
|
|
}
|
|
}
|
|
|
|
void Screens::setCount(int count)
|
|
{
|
|
if (m_count == count) {
|
|
return;
|
|
}
|
|
const int previous = m_count;
|
|
m_count = count;
|
|
emit countChanged(previous, count);
|
|
}
|
|
|
|
void Screens::setCurrent(int current)
|
|
{
|
|
if (m_current == current) {
|
|
return;
|
|
}
|
|
m_current = current;
|
|
emit currentChanged();
|
|
}
|
|
|
|
void Screens::setCurrent(const QPoint &pos)
|
|
{
|
|
setCurrent(number(pos));
|
|
}
|
|
|
|
void Screens::setCurrent(const Client *c)
|
|
{
|
|
if (!c->isActive()) {
|
|
return;
|
|
}
|
|
if (!c->isOnScreen(m_current)) {
|
|
setCurrent(c->screen());
|
|
}
|
|
}
|
|
|
|
void Screens::setCurrentFollowsMouse(bool follows)
|
|
{
|
|
if (m_currentFollowsMouse == follows) {
|
|
return;
|
|
}
|
|
m_currentFollowsMouse = follows;
|
|
}
|
|
|
|
int Screens::current() const
|
|
{
|
|
if (m_currentFollowsMouse) {
|
|
return number(Cursor::pos());
|
|
}
|
|
Client *client = Workspace::self()->activeClient();
|
|
if (client && !client->isOnScreen(m_current)) {
|
|
return client->screen();
|
|
}
|
|
return m_current;
|
|
}
|
|
|
|
int Screens::intersecting(const QRect &r) const
|
|
{
|
|
int cnt = 0;
|
|
for (int i = 0; i < count(); ++i) {
|
|
if (geometry(i).intersects(r)) {
|
|
++cnt;
|
|
}
|
|
}
|
|
return cnt;
|
|
}
|
|
|
|
DesktopWidgetScreens::DesktopWidgetScreens(QObject *parent)
|
|
: Screens(parent)
|
|
, m_desktop(QApplication::desktop())
|
|
{
|
|
}
|
|
|
|
DesktopWidgetScreens::~DesktopWidgetScreens()
|
|
{
|
|
}
|
|
|
|
void DesktopWidgetScreens::init()
|
|
{
|
|
Screens::init();
|
|
connect(m_desktop, SIGNAL(screenCountChanged(int)), SLOT(startChangedTimer()));
|
|
connect(m_desktop, SIGNAL(resized(int)), SLOT(startChangedTimer()));
|
|
updateCount();
|
|
}
|
|
|
|
QRect DesktopWidgetScreens::geometry(int screen) const
|
|
{
|
|
if (Screens::self()->isChanging())
|
|
const_cast<DesktopWidgetScreens*>(this)->updateCount();
|
|
return m_desktop->screenGeometry(screen);
|
|
}
|
|
|
|
QSize DesktopWidgetScreens::size(int screen) const
|
|
{
|
|
return geometry(screen).size();
|
|
}
|
|
|
|
int DesktopWidgetScreens::number(const QPoint &pos) const
|
|
{
|
|
if (Screens::self()->isChanging())
|
|
const_cast<DesktopWidgetScreens*>(this)->updateCount();
|
|
return m_desktop->screenNumber(pos);
|
|
}
|
|
|
|
void DesktopWidgetScreens::updateCount()
|
|
{
|
|
setCount(m_desktop->screenCount());
|
|
}
|
|
|
|
} // namespace
|