mirror of
https://github.com/KDE/latte-dock.git
synced 2024-12-24 17:33:50 +03:00
4d3b5e86e8
--introduce also windows whitelist that can be used in order to force tracking in some windows
333 lines
9.2 KiB
C++
333 lines
9.2 KiB
C++
/*
|
|
* Copyright 2016 Smith AR <audoban@openmailbox.org>
|
|
* Michail Vourlakos <mvourlakos@gmail.com>
|
|
*
|
|
* This file is part of Latte-Dock
|
|
*
|
|
* Latte-Dock 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.
|
|
*
|
|
* Latte-Dock 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 "abstractwindowinterface.h"
|
|
|
|
// local
|
|
#include "tracker/schemes.h"
|
|
#include "tracker/windowstracker.h"
|
|
#include "../lattecorona.h"
|
|
|
|
// Qt
|
|
#include <QDebug>
|
|
|
|
// KDE
|
|
#include <KActivities/Controller>
|
|
|
|
namespace Latte {
|
|
namespace WindowSystem {
|
|
|
|
#define MAXPLASMAPANELTHICKNESS 96
|
|
#define MAXSIDEPANELTHICKNESS 512
|
|
|
|
AbstractWindowInterface::AbstractWindowInterface(QObject *parent)
|
|
: QObject(parent)
|
|
{
|
|
m_activities = new KActivities::Consumer(this);
|
|
m_currentActivity = m_activities->currentActivity();
|
|
|
|
m_corona = qobject_cast<Latte::Corona *>(parent);
|
|
m_windowsTracker = new Tracker::Windows(this);
|
|
m_schemesTracker = new Tracker::Schemes(this);
|
|
|
|
rulesConfig = KSharedConfig::openConfig(QStringLiteral("taskmanagerrulesrc"));
|
|
|
|
m_windowWaitingTimer.setInterval(150);
|
|
m_windowWaitingTimer.setSingleShot(true);
|
|
|
|
connect(&m_windowWaitingTimer, &QTimer::timeout, this, [&]() {
|
|
WindowId wid = m_windowChangedWaiting;
|
|
m_windowChangedWaiting = QVariant();
|
|
emit windowChanged(wid);
|
|
});
|
|
|
|
connect(this, &AbstractWindowInterface::windowRemoved, this, &AbstractWindowInterface::windowRemovedSlot);
|
|
|
|
// connect(this, &AbstractWindowInterface::windowChanged, this, [&](WindowId wid) {
|
|
// qDebug() << "WINDOW CHANGED ::: " << wid;
|
|
// });
|
|
|
|
connect(m_activities.data(), &KActivities::Consumer::currentActivityChanged, this, [&](const QString &id) {
|
|
m_currentActivity = id;
|
|
emit currentActivityChanged();
|
|
});
|
|
|
|
}
|
|
|
|
AbstractWindowInterface::~AbstractWindowInterface()
|
|
{
|
|
m_windowWaitingTimer.stop();
|
|
|
|
m_schemesTracker->deleteLater();
|
|
m_windowsTracker->deleteLater();
|
|
}
|
|
|
|
QString AbstractWindowInterface::currentDesktop()
|
|
{
|
|
return m_currentDesktop;
|
|
}
|
|
|
|
QString AbstractWindowInterface::currentActivity()
|
|
{
|
|
return m_currentActivity;
|
|
}
|
|
|
|
Latte::Corona *AbstractWindowInterface::corona()
|
|
{
|
|
return m_corona;
|
|
}
|
|
|
|
Tracker::Schemes *AbstractWindowInterface::schemesTracker()
|
|
{
|
|
return m_schemesTracker;
|
|
}
|
|
|
|
Tracker::Windows *AbstractWindowInterface::windowsTracker() const
|
|
{
|
|
return m_windowsTracker;
|
|
}
|
|
|
|
bool AbstractWindowInterface::isIgnored(const WindowId &wid) const
|
|
{
|
|
return m_ignoredWindows.contains(wid);
|
|
}
|
|
|
|
bool AbstractWindowInterface::isFullScreenWindow(const QRect &wGeometry) const
|
|
{
|
|
if (wGeometry.isEmpty()) {
|
|
return false;
|
|
}
|
|
|
|
for (const auto scr : qGuiApp->screens()) {
|
|
if (wGeometry == scr->geometry()) {
|
|
return true;
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
bool AbstractWindowInterface::isPlasmaPanel(const QRect &wGeometry) const
|
|
{
|
|
if (wGeometry.isEmpty()) {
|
|
return false;
|
|
}
|
|
|
|
bool isTouchingHorizontalEdge{false};
|
|
bool isTouchingVerticalEdge{false};
|
|
|
|
for (const auto scr : qGuiApp->screens()) {
|
|
if (scr->geometry().contains(wGeometry.center())) {
|
|
if (wGeometry.y() == scr->geometry().y() || wGeometry.bottom() == scr->geometry().bottom()) {
|
|
isTouchingHorizontalEdge = true;
|
|
}
|
|
|
|
if (wGeometry.left() == scr->geometry().left() || wGeometry.right() == scr->geometry().right()) {
|
|
isTouchingVerticalEdge = true;
|
|
}
|
|
|
|
if (isTouchingVerticalEdge && isTouchingHorizontalEdge) {
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
if ((isTouchingHorizontalEdge && wGeometry.height() < MAXPLASMAPANELTHICKNESS)
|
|
|| (isTouchingVerticalEdge && wGeometry.width() < MAXPLASMAPANELTHICKNESS)) {
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
bool AbstractWindowInterface::isSidepanel(const QRect &wGeometry) const
|
|
{
|
|
bool isVertical = wGeometry.height() > wGeometry.width();
|
|
|
|
int thickness = qMin(wGeometry.width(), wGeometry.height());
|
|
int length = qMax(wGeometry.width(), wGeometry.height());
|
|
|
|
QRect screenGeometry;
|
|
|
|
for (const auto scr : qGuiApp->screens()) {
|
|
if (scr->geometry().contains(wGeometry.center())) {
|
|
screenGeometry = scr->geometry();
|
|
break;
|
|
}
|
|
}
|
|
|
|
bool thicknessIsAcccepted = isVertical && ((thickness > MAXPLASMAPANELTHICKNESS) && (thickness < MAXSIDEPANELTHICKNESS));
|
|
bool lengthIsAccepted = isVertical && !screenGeometry.isEmpty() && (length > 0.6 * screenGeometry.height());
|
|
float sideRatio = (float)wGeometry.width() / (float)wGeometry.height();
|
|
|
|
return (thicknessIsAcccepted && lengthIsAccepted && sideRatio<0.4);
|
|
}
|
|
|
|
bool AbstractWindowInterface::hasBlockedTracking(const WindowId &wid) const
|
|
{
|
|
return (!isWhitelistedWindow(wid) && (isRegisteredPlasmaIgnoredWindow(wid) || isIgnored(wid)));
|
|
}
|
|
|
|
bool AbstractWindowInterface::isRegisteredPlasmaIgnoredWindow(const WindowId &wid) const
|
|
{
|
|
return m_plasmaIgnoredWindows.contains(wid);
|
|
}
|
|
|
|
bool AbstractWindowInterface::isWhitelistedWindow(const WindowId &wid) const
|
|
{
|
|
return m_whitelistedWindows.contains(wid);
|
|
}
|
|
|
|
bool AbstractWindowInterface::inCurrentDesktopActivity(const WindowInfoWrap &winfo)
|
|
{
|
|
return (winfo.isValid() && winfo.isOnDesktop(currentDesktop()) && winfo.isOnActivity(currentActivity()));
|
|
}
|
|
|
|
//! Register Latte Ignored Windows in order to NOT be tracked
|
|
void AbstractWindowInterface::registerIgnoredWindow(WindowId wid)
|
|
{
|
|
if (!wid.isNull() && !m_ignoredWindows.contains(wid)) {
|
|
m_ignoredWindows.append(wid);
|
|
emit windowChanged(wid);
|
|
}
|
|
}
|
|
|
|
void AbstractWindowInterface::unregisterIgnoredWindow(WindowId wid)
|
|
{
|
|
if (m_ignoredWindows.contains(wid)) {
|
|
m_ignoredWindows.removeAll(wid);
|
|
emit windowRemoved(wid);
|
|
}
|
|
}
|
|
|
|
void AbstractWindowInterface::registerPlasmaIgnoredWindow(WindowId wid)
|
|
{
|
|
if (!wid.isNull() && !m_plasmaIgnoredWindows.contains(wid)) {
|
|
m_plasmaIgnoredWindows.append(wid);
|
|
emit windowChanged(wid);
|
|
}
|
|
}
|
|
|
|
void AbstractWindowInterface::unregisterPlasmaIgnoredWindow(WindowId wid)
|
|
{
|
|
if (m_plasmaIgnoredWindows.contains(wid)) {
|
|
m_plasmaIgnoredWindows.removeAll(wid);
|
|
}
|
|
}
|
|
|
|
void AbstractWindowInterface::registerWhitelistedWindow(WindowId wid)
|
|
{
|
|
if (!wid.isNull() && !m_whitelistedWindows.contains(wid)) {
|
|
m_whitelistedWindows.append(wid);
|
|
emit windowChanged(wid);
|
|
}
|
|
}
|
|
|
|
void AbstractWindowInterface::unregisterWhitelistedWindow(WindowId wid)
|
|
{
|
|
if (m_whitelistedWindows.contains(wid)) {
|
|
m_whitelistedWindows.removeAll(wid);
|
|
}
|
|
}
|
|
|
|
void AbstractWindowInterface::windowRemovedSlot(WindowId wid)
|
|
{
|
|
if (m_plasmaIgnoredWindows.contains(wid)) {
|
|
unregisterPlasmaIgnoredWindow(wid);
|
|
}
|
|
|
|
if (m_ignoredWindows.contains(wid)) {
|
|
unregisterIgnoredWindow(wid);
|
|
}
|
|
|
|
if (m_whitelistedWindows.contains(wid)) {
|
|
unregisterWhitelistedWindow(wid);
|
|
}
|
|
}
|
|
|
|
//! Activities switching
|
|
void AbstractWindowInterface::switchToNextActivity()
|
|
{
|
|
QStringList runningActivities = m_activities->activities(KActivities::Info::State::Running);
|
|
if (runningActivities.count() <= 1) {
|
|
return;
|
|
}
|
|
|
|
int curPos = runningActivities.indexOf(m_currentActivity);
|
|
int nextPos = curPos + 1;
|
|
|
|
if (curPos == runningActivities.count() -1) {
|
|
nextPos = 0;
|
|
}
|
|
|
|
KActivities::Controller activitiesController;
|
|
activitiesController.setCurrentActivity(runningActivities.at(nextPos));
|
|
}
|
|
|
|
void AbstractWindowInterface::switchToPreviousActivity()
|
|
{
|
|
QStringList runningActivities = m_activities->activities(KActivities::Info::State::Running);
|
|
if (runningActivities.count() <= 1) {
|
|
return;
|
|
}
|
|
|
|
int curPos = runningActivities.indexOf(m_currentActivity);
|
|
int nextPos = curPos - 1;
|
|
|
|
if (curPos == 0) {
|
|
nextPos = runningActivities.count() - 1;
|
|
}
|
|
|
|
KActivities::Controller activitiesController;
|
|
activitiesController.setCurrentActivity(runningActivities.at(nextPos));
|
|
}
|
|
|
|
//! Delay window changed trigerring
|
|
void AbstractWindowInterface::considerWindowChanged(WindowId wid)
|
|
{
|
|
//! Consider if the windowChanged signal should be sent DIRECTLY or WAIT
|
|
|
|
if (m_windowChangedWaiting == wid && m_windowWaitingTimer.isActive()) {
|
|
//! window should be sent later
|
|
m_windowWaitingTimer.start();
|
|
return;
|
|
}
|
|
|
|
if (m_windowChangedWaiting != wid && !m_windowWaitingTimer.isActive()) {
|
|
//! window should be sent later
|
|
m_windowChangedWaiting = wid;
|
|
m_windowWaitingTimer.start();
|
|
}
|
|
|
|
if (m_windowChangedWaiting != wid && m_windowWaitingTimer.isActive()) {
|
|
m_windowWaitingTimer.stop();
|
|
//! sent previous waiting window
|
|
emit windowChanged(m_windowChangedWaiting);
|
|
|
|
//! retrigger waiting for the upcoming window
|
|
m_windowChangedWaiting = wid;
|
|
m_windowWaitingTimer.start();
|
|
}
|
|
}
|
|
|
|
}
|
|
}
|
|
|