mirror of
https://github.com/KDE/latte-dock.git
synced 2025-01-10 21:18:19 +03:00
1f0acbc3b5
--by default now Latte supports the KWin Edges mechanism, this way the following issues are fixed: fix #935, fix #721, fix #687, fix #512, fix #514 --improve dock behavior when the user user touch gestures --fix #144, support KWin bright placeholders --this behavior can be disabled if the user prefers the old one from the Tweaks page --in !compositing no more is shown a 1px line at the screen edge when the dock is hidden --support KWin Edges in Multiple Layouts state under X11
392 lines
11 KiB
C++
392 lines
11 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 "waylandinterface.h"
|
|
#include "dockcorona.h"
|
|
#include "dock/dockview.h"
|
|
#include "dock/screenedgeghostwindow.h"
|
|
#include "../liblattedock/extras.h"
|
|
|
|
#include <QDebug>
|
|
#include <QTimer>
|
|
#include <QApplication>
|
|
#include <QSignalMapper>
|
|
#include <QtX11Extras/QX11Info>
|
|
#include <QRasterWindow>
|
|
|
|
#include <KWindowSystem>
|
|
#include <KWindowInfo>
|
|
#include <NETWM>
|
|
|
|
#include <KWayland/Client/surface.h>
|
|
|
|
using namespace KWayland::Client;
|
|
|
|
namespace Latte {
|
|
|
|
class Private::GhostWindow : public QRasterWindow {
|
|
Q_OBJECT
|
|
|
|
public:
|
|
GhostWindow(WaylandInterface *waylandInterface)
|
|
: m_waylandInterface(waylandInterface) {
|
|
setFlags(Qt::FramelessWindowHint
|
|
| Qt::WindowStaysOnTopHint
|
|
| Qt::NoDropShadowWindowHint
|
|
| Qt::WindowDoesNotAcceptFocus);
|
|
|
|
setupWaylandIntegration();
|
|
show();
|
|
}
|
|
|
|
~GhostWindow() {
|
|
delete m_shellSurface;
|
|
}
|
|
|
|
void setGeometry(const QRect &rect) {
|
|
QWindow::setGeometry(rect);
|
|
setMaximumSize(rect.size());
|
|
m_shellSurface->setPosition(rect.topLeft());
|
|
}
|
|
|
|
void setupWaylandIntegration() {
|
|
using namespace KWayland::Client;
|
|
|
|
if (m_shellSurface)
|
|
return;
|
|
|
|
Surface *s{Surface::fromWindow(this)};
|
|
|
|
if (!s)
|
|
return;
|
|
|
|
m_shellSurface = m_waylandInterface->waylandDockCoronaInterface()->createSurface(s, this);
|
|
qDebug() << "wayland ghost window surface was created...";
|
|
|
|
m_shellSurface->setSkipTaskbar(true);
|
|
m_shellSurface->setPanelTakesFocus(false);
|
|
m_shellSurface->setRole(PlasmaShellSurface::Role::Panel);
|
|
m_shellSurface->setPanelBehavior(PlasmaShellSurface::PanelBehavior::AlwaysVisible);
|
|
}
|
|
|
|
KWayland::Client::PlasmaShellSurface *m_shellSurface{nullptr};
|
|
WaylandInterface *m_waylandInterface{nullptr};
|
|
};
|
|
|
|
WaylandInterface::WaylandInterface(QObject *parent)
|
|
: AbstractWindowInterface(parent)
|
|
{
|
|
m_corona = qobject_cast<DockCorona *>(parent);
|
|
|
|
m_activities = new KActivities::Consumer(this);
|
|
|
|
|
|
connect(m_activities.data(), &KActivities::Consumer::currentActivityChanged
|
|
, this, &WaylandInterface::currentActivityChanged);
|
|
}
|
|
|
|
WaylandInterface::~WaylandInterface()
|
|
{
|
|
}
|
|
|
|
void WaylandInterface::init()
|
|
{
|
|
}
|
|
|
|
void WaylandInterface::initWindowManagement(KWayland::Client::PlasmaWindowManagement *windowManagement)
|
|
{
|
|
m_windowManagement = windowManagement;
|
|
|
|
connect(m_windowManagement, &PlasmaWindowManagement::windowCreated, this, &WaylandInterface::windowCreatedProxy);
|
|
connect(m_windowManagement, &PlasmaWindowManagement::activeWindowChanged, this, [&]() noexcept {
|
|
auto w = m_windowManagement->activeWindow();
|
|
emit activeWindowChanged(w ? w->internalId() : 0);
|
|
}, Qt::QueuedConnection);
|
|
}
|
|
|
|
KWayland::Client::PlasmaShell *WaylandInterface::waylandDockCoronaInterface() const
|
|
{
|
|
return m_corona->waylandDockCoronaInterface();
|
|
}
|
|
|
|
void WaylandInterface::setDockExtraFlags(QWindow &view)
|
|
{
|
|
Q_UNUSED(view)
|
|
}
|
|
|
|
void WaylandInterface::setDockStruts(QWindow &view, const QRect &rect , Plasma::Types::Location location)
|
|
{
|
|
if (!m_ghostWindows.contains(view.winId()))
|
|
m_ghostWindows[view.winId()] = new Private::GhostWindow(this);
|
|
|
|
auto w = m_ghostWindows[view.winId()];
|
|
|
|
switch (location) {
|
|
case Plasma::Types::TopEdge:
|
|
case Plasma::Types::BottomEdge:
|
|
w->setGeometry({rect.x() + rect.width() / 2, rect.y(), 1, rect.height()});
|
|
break;
|
|
|
|
case Plasma::Types::LeftEdge:
|
|
case Plasma::Types::RightEdge:
|
|
w->setGeometry({rect.x(), rect.y() + rect.height() / 2, rect.width(), 1});
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
|
|
}
|
|
|
|
void WaylandInterface::setWindowOnActivities(QWindow &window, const QStringList &activities)
|
|
{
|
|
//! needs to updated to wayland case
|
|
// KWindowSystem::setOnActivities(view.winId(), activities);
|
|
}
|
|
|
|
void WaylandInterface::removeDockStruts(QWindow &view) const
|
|
{
|
|
delete m_ghostWindows.take(view.winId());
|
|
}
|
|
|
|
WindowId WaylandInterface::activeWindow() const
|
|
{
|
|
if (!m_windowManagement) {
|
|
return 0;
|
|
}
|
|
|
|
auto wid = m_windowManagement->activeWindow();
|
|
|
|
return wid ? wid->internalId() : 0;
|
|
}
|
|
|
|
const std::list<WindowId> &WaylandInterface::windows() const
|
|
{
|
|
return m_windows;
|
|
}
|
|
|
|
void WaylandInterface::setKeepAbove(const QDialog &dialog, bool above) const
|
|
{
|
|
if (above) {
|
|
KWindowSystem::setState(dialog.winId(), NET::KeepAbove);
|
|
} else {
|
|
KWindowSystem::clearState(dialog.winId(), NET::KeepAbove);
|
|
}
|
|
}
|
|
|
|
void WaylandInterface::skipTaskBar(const QDialog &dialog) const
|
|
{
|
|
KWindowSystem::setState(dialog.winId(), NET::SkipTaskbar);
|
|
}
|
|
|
|
void WaylandInterface::slideWindow(QWindow &view, AbstractWindowInterface::Slide location) const
|
|
{
|
|
auto slideLocation = KWindowEffects::NoEdge;
|
|
|
|
switch (location) {
|
|
case Slide::Top:
|
|
slideLocation = KWindowEffects::TopEdge;
|
|
break;
|
|
|
|
case Slide::Bottom:
|
|
slideLocation = KWindowEffects::BottomEdge;
|
|
break;
|
|
|
|
case Slide::Left:
|
|
slideLocation = KWindowEffects::LeftEdge;
|
|
break;
|
|
|
|
case Slide::Right:
|
|
slideLocation = KWindowEffects::RightEdge;
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
|
|
KWindowEffects::slideWindow(view.winId(), slideLocation, -1);
|
|
}
|
|
|
|
void WaylandInterface::enableBlurBehind(QWindow &view) const
|
|
{
|
|
KWindowEffects::enableBlurBehind(view.winId());
|
|
}
|
|
|
|
void WaylandInterface::setEdgeStateFor(QWindow *view, bool active) const
|
|
{
|
|
ScreenEdgeGhostWindow *window = qobject_cast<ScreenEdgeGhostWindow *>(view);
|
|
|
|
if (!window) {
|
|
return;
|
|
}
|
|
|
|
if (window->parentDock()->surface() && window->parentDock()->visibility()
|
|
&& (window->parentDock()->visibility()->mode() == Dock::DodgeActive
|
|
|| window->parentDock()->visibility()->mode() == Dock::DodgeMaximized
|
|
|| window->parentDock()->visibility()->mode() == Dock::DodgeAllWindows
|
|
|| window->parentDock()->visibility()->mode() == Dock::AutoHide)) {
|
|
if (active) {
|
|
window->showWithMask();
|
|
window->surface()->requestHideAutoHidingPanel();
|
|
} else {
|
|
window->hideWithMask();
|
|
window->surface()->requestShowAutoHidingPanel();
|
|
}
|
|
}
|
|
}
|
|
|
|
WindowInfoWrap WaylandInterface::requestInfoActive() const
|
|
{
|
|
if (!m_windowManagement) {
|
|
return {};
|
|
}
|
|
|
|
auto w = m_windowManagement->activeWindow();
|
|
|
|
if (!w) return {};
|
|
|
|
WindowInfoWrap winfoWrap;
|
|
|
|
winfoWrap.setIsValid(true);
|
|
|
|
winfoWrap.setWid(w->internalId());
|
|
|
|
winfoWrap.setIsActive(w->isActive());
|
|
|
|
winfoWrap.setIsMinimized(w->isMinimized());
|
|
|
|
winfoWrap.setIsMaxVert(w->isMaximized());
|
|
|
|
winfoWrap.setIsMaxHoriz(w->isMaximized());
|
|
|
|
winfoWrap.setIsFullscreen(w->isFullscreen());
|
|
|
|
winfoWrap.setIsShaded(w->isShaded());
|
|
|
|
winfoWrap.setGeometry(w->geometry());
|
|
|
|
winfoWrap.setIsKeepAbove(w->isKeepAbove());
|
|
|
|
return winfoWrap;
|
|
}
|
|
|
|
bool WaylandInterface::isOnCurrentDesktop(WindowId wid) const
|
|
{
|
|
if (!m_windowManagement) {
|
|
return false;
|
|
}
|
|
|
|
auto it = std::find_if(m_windowManagement->windows().constBegin(), m_windowManagement->windows().constEnd(), [&wid](PlasmaWindow * w) noexcept {
|
|
return w->isValid() && w->internalId() == wid;
|
|
});
|
|
|
|
//qDebug() << "desktop:" << (it != m_windowManagement->windows().constEnd() ? (*it)->virtualDesktop() : -1) << KWindowSystem::currentDesktop();
|
|
//return true;
|
|
return it != m_windowManagement->windows().constEnd() && ((*it)->virtualDesktop() == KWindowSystem::currentDesktop() || (*it)->isOnAllDesktops());
|
|
}
|
|
|
|
bool WaylandInterface::isOnCurrentActivity(WindowId wid) const
|
|
{
|
|
auto it = std::find_if(m_windowManagement->windows().constBegin(), m_windowManagement->windows().constEnd(), [&wid](PlasmaWindow * w) noexcept {
|
|
return w->isValid() && w->internalId() == wid;
|
|
});
|
|
|
|
//TODO: Not yet implemented
|
|
return it != m_windowManagement->windows().constEnd() && true;
|
|
|
|
}
|
|
|
|
WindowInfoWrap WaylandInterface::requestInfo(WindowId wid) const
|
|
{
|
|
auto it = std::find_if(m_windowManagement->windows().constBegin(), m_windowManagement->windows().constEnd(), [&wid](PlasmaWindow * w) noexcept {
|
|
return w->isValid() && w->internalId() == wid;
|
|
});
|
|
|
|
if (it == m_windowManagement->windows().constEnd())
|
|
return {};
|
|
|
|
WindowInfoWrap winfoWrap;
|
|
|
|
auto w = *it;
|
|
|
|
if (isValidWindow(w)) {
|
|
winfoWrap.setIsValid(true);
|
|
winfoWrap.setWid(wid);
|
|
winfoWrap.setIsActive(w->isActive());
|
|
winfoWrap.setIsMinimized(w->isMinimized());
|
|
winfoWrap.setIsMaxVert(w->isMaximized());
|
|
winfoWrap.setIsMaxHoriz(w->isMaximized());
|
|
winfoWrap.setIsFullscreen(w->isFullscreen());
|
|
winfoWrap.setIsShaded(w->isShaded());
|
|
winfoWrap.setGeometry(w->geometry());
|
|
} else if (w->appId() == QLatin1String("org.kde.plasmashell")) {
|
|
winfoWrap.setIsValid(true);
|
|
winfoWrap.setIsPlasmaDesktop(true);
|
|
winfoWrap.setWid(wid);
|
|
}
|
|
|
|
return winfoWrap;
|
|
}
|
|
|
|
|
|
inline bool WaylandInterface::isValidWindow(const KWayland::Client::PlasmaWindow *w) const
|
|
{
|
|
return w->isValid() && !w->skipTaskbar();
|
|
}
|
|
|
|
void WaylandInterface::windowCreatedProxy(KWayland::Client::PlasmaWindow *w)
|
|
{
|
|
if (!isValidWindow(w)) return;
|
|
|
|
if (!mapper) mapper = new QSignalMapper(this);
|
|
|
|
mapper->setMapping(w, w);
|
|
|
|
connect(w, &PlasmaWindow::unmapped, this, [ &, win = w]() noexcept {
|
|
mapper->removeMappings(win);
|
|
m_windows.remove(win->internalId());
|
|
emit windowRemoved(win->internalId());
|
|
});
|
|
|
|
connect(w, SIGNAL(activeChanged()), mapper, SLOT(map()));
|
|
connect(w, SIGNAL(fullscreenChanged()), mapper, SLOT(map()));
|
|
connect(w, SIGNAL(geometryChanged()), mapper, SLOT(map()));
|
|
connect(w, SIGNAL(maximizedChanged()), mapper, SLOT(map()));
|
|
connect(w, SIGNAL(minimizedChanged()), mapper, SLOT(map()));
|
|
connect(w, SIGNAL(shadedChanged()), mapper, SLOT(map()));
|
|
connect(w, SIGNAL(skipTaskbarChanged()), mapper, SLOT(map()));
|
|
connect(w, SIGNAL(onAllDesktopsChanged()), mapper, SLOT(map()));
|
|
connect(w, SIGNAL(virtualDesktopChanged()), mapper, SLOT(map()));
|
|
|
|
connect(mapper, static_cast<void (QSignalMapper::*)(QObject *)>(&QSignalMapper::mapped)
|
|
, this, [&](QObject * w) noexcept {
|
|
//qDebug() << "window changed:" << qobject_cast<PlasmaWindow *>(w)->appId();
|
|
emit windowChanged(qobject_cast<PlasmaWindow *>(w)->internalId());
|
|
});
|
|
|
|
m_windows.push_back(w->internalId());
|
|
|
|
emit windowAdded(w->internalId());
|
|
}
|
|
|
|
}
|
|
|
|
#include "waylandinterface.moc"
|