2009-01-31 15:12:14 +00:00
|
|
|
/********************************************************************
|
|
|
|
KWin - the KDE window manager
|
|
|
|
This file is part of the KDE project.
|
|
|
|
|
|
|
|
Copyright (C) 2009 Lucas Murray <lmurray@undefinedfire.com>
|
|
|
|
|
|
|
|
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 "highlightwindow.h"
|
|
|
|
|
|
|
|
namespace KWin
|
|
|
|
{
|
|
|
|
|
|
|
|
HighlightWindowEffect::HighlightWindowEffect()
|
2011-01-30 14:34:42 +00:00
|
|
|
: m_finishing(false)
|
2011-12-13 13:28:20 +00:00
|
|
|
, m_fadeDuration(float(animationTime(150)))
|
2011-01-30 14:34:42 +00:00
|
|
|
, m_monitorWindow(NULL)
|
|
|
|
{
|
2012-12-13 23:09:47 +00:00
|
|
|
m_atom = effects->announceSupportProperty("_KDE_WINDOW_HIGHLIGHT", this);
|
2019-01-01 20:48:53 +00:00
|
|
|
connect(effects, &EffectsHandler::windowAdded, this, &HighlightWindowEffect::slotWindowAdded);
|
|
|
|
connect(effects, &EffectsHandler::windowClosed, this, &HighlightWindowEffect::slotWindowClosed);
|
|
|
|
connect(effects, &EffectsHandler::windowDeleted, this, &HighlightWindowEffect::slotWindowDeleted);
|
|
|
|
connect(effects, &EffectsHandler::propertyNotify, this,
|
|
|
|
[this](EffectWindow *w, long atom) {
|
|
|
|
slotPropertyNotify(w, atom, nullptr);
|
|
|
|
}
|
|
|
|
);
|
2017-09-10 14:51:18 +00:00
|
|
|
connect(effects, &EffectsHandler::xcbConnectionChanged, this,
|
|
|
|
[this] {
|
|
|
|
m_atom = effects->announceSupportProperty("_KDE_WINDOW_HIGHLIGHT", this);
|
|
|
|
}
|
|
|
|
);
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2009-01-31 15:12:14 +00:00
|
|
|
|
|
|
|
HighlightWindowEffect::~HighlightWindowEffect()
|
2011-01-30 14:34:42 +00:00
|
|
|
{
|
|
|
|
}
|
2009-01-31 15:12:14 +00:00
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
static bool isInitiallyHidden(EffectWindow* w)
|
|
|
|
{
|
|
|
|
// Is the window initially hidden until it is highlighted?
|
2012-01-12 06:42:55 +00:00
|
|
|
return w->isMinimized() || !w->isCurrentTab() || !w->isOnCurrentDesktop();
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2009-12-20 12:48:54 +00:00
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
void HighlightWindowEffect::prePaintWindow(EffectWindow* w, WindowPrePaintData& data, int time)
|
|
|
|
{
|
2009-02-03 10:51:36 +00:00
|
|
|
// Calculate window opacities
|
2011-12-13 13:28:20 +00:00
|
|
|
QHash<EffectWindow*, float>::iterator opacity = m_windowOpacity.find(w);
|
2011-01-30 14:34:42 +00:00
|
|
|
if (!m_highlightedWindows.isEmpty()) {
|
|
|
|
// Initial fade out and changing highlight animation
|
2011-12-13 13:28:20 +00:00
|
|
|
if (opacity == m_windowOpacity.end())
|
|
|
|
opacity = m_windowOpacity.insertMulti(w, 0.0f);
|
|
|
|
float oldOpacity = *opacity;
|
2011-01-30 14:34:42 +00:00
|
|
|
if (m_highlightedWindows.contains(w))
|
2011-12-13 13:28:20 +00:00
|
|
|
*opacity = qMin(1.0f, oldOpacity + time / m_fadeDuration);
|
2011-01-30 14:34:42 +00:00
|
|
|
else if (w->isNormalWindow() || w->isDialog()) // Only fade out windows
|
2011-12-13 13:28:20 +00:00
|
|
|
*opacity = qMax(isInitiallyHidden(w) ? 0.0f : 0.15f, oldOpacity - time / m_fadeDuration);
|
2009-02-03 10:51:36 +00:00
|
|
|
|
2011-12-13 13:28:20 +00:00
|
|
|
if (*opacity < 0.98f)
|
2009-02-03 10:51:36 +00:00
|
|
|
data.setTranslucent();
|
2011-12-13 13:28:20 +00:00
|
|
|
if (oldOpacity != *opacity)
|
2016-09-12 09:13:11 +00:00
|
|
|
effects->addRepaint(w->expandedGeometry());
|
2011-01-30 14:34:42 +00:00
|
|
|
} else if (m_finishing && m_windowOpacity.contains(w)) {
|
|
|
|
// Final fading back in animation
|
2011-12-13 13:28:20 +00:00
|
|
|
if (opacity == m_windowOpacity.end())
|
|
|
|
opacity = m_windowOpacity.insert(w, 0.0f);
|
|
|
|
float oldOpacity = *opacity;
|
2011-01-30 14:34:42 +00:00
|
|
|
if (isInitiallyHidden(w))
|
2011-12-13 13:28:20 +00:00
|
|
|
*opacity = qMax(0.0f, oldOpacity - time / m_fadeDuration);
|
2009-12-20 12:48:54 +00:00
|
|
|
else
|
2011-12-13 13:28:20 +00:00
|
|
|
*opacity = qMin(1.0f, oldOpacity + time / m_fadeDuration);
|
2009-02-03 10:51:36 +00:00
|
|
|
|
2011-12-13 13:28:20 +00:00
|
|
|
if (*opacity < 0.98f)
|
2009-02-03 10:51:36 +00:00
|
|
|
data.setTranslucent();
|
2011-12-13 13:28:20 +00:00
|
|
|
if (oldOpacity != *opacity)
|
2016-09-12 09:13:11 +00:00
|
|
|
effects->addRepaint(w->expandedGeometry());
|
2009-02-03 10:51:36 +00:00
|
|
|
|
2015-01-17 22:48:08 +00:00
|
|
|
if (*opacity > 0.98f || *opacity < 0.02f) {
|
2011-01-30 14:34:42 +00:00
|
|
|
m_windowOpacity.remove(w); // We default to 1.0
|
2015-01-17 22:48:08 +00:00
|
|
|
opacity = m_windowOpacity.end();
|
|
|
|
}
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2009-02-03 10:51:36 +00:00
|
|
|
|
2009-12-20 12:48:54 +00:00
|
|
|
// Show tabbed windows and windows on other desktops if highlighted
|
2011-12-13 13:28:20 +00:00
|
|
|
if (opacity != m_windowOpacity.end() && *opacity > 0.01) {
|
|
|
|
if (w->isMinimized())
|
|
|
|
w->enablePainting(EffectWindow::PAINT_DISABLED_BY_MINIMIZE);
|
2012-01-12 06:42:55 +00:00
|
|
|
if (!w->isCurrentTab())
|
|
|
|
w->enablePainting(EffectWindow::PAINT_DISABLED_BY_TAB_GROUP);
|
2011-01-30 14:34:42 +00:00
|
|
|
if (!w->isOnCurrentDesktop())
|
|
|
|
w->enablePainting(EffectWindow::PAINT_DISABLED_BY_DESKTOP);
|
2009-01-31 15:12:14 +00:00
|
|
|
}
|
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
effects->prePaintWindow(w, data, time);
|
|
|
|
}
|
|
|
|
|
|
|
|
void HighlightWindowEffect::paintWindow(EffectWindow* w, int mask, QRegion region, WindowPaintData& data)
|
|
|
|
{
|
2012-07-12 15:20:17 +00:00
|
|
|
data.multiplyOpacity(m_windowOpacity.value(w, 1.0f));
|
2011-01-30 14:34:42 +00:00
|
|
|
effects->paintWindow(w, mask, region, data);
|
|
|
|
}
|
2009-01-31 15:12:14 +00:00
|
|
|
|
2011-02-25 21:06:02 +00:00
|
|
|
void HighlightWindowEffect::slotWindowAdded(EffectWindow* w)
|
2011-01-30 14:34:42 +00:00
|
|
|
{
|
|
|
|
if (!m_highlightedWindows.isEmpty()) {
|
|
|
|
// The effect is activated thus we need to add it to the opacity hash
|
2013-09-03 19:55:39 +00:00
|
|
|
foreach (const WId id, m_highlightedIds) {
|
|
|
|
if (w == effects->findWindow(id)) {
|
|
|
|
m_windowOpacity[w] = 1.0; // this window was demanded to be highlighted before it appeared
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
m_windowOpacity[w] = 0.15; // this window is not currently highlighted
|
2009-01-31 15:12:14 +00:00
|
|
|
}
|
2013-09-03 19:55:39 +00:00
|
|
|
slotPropertyNotify(w, m_atom, w); // Check initial value
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2009-01-31 15:12:14 +00:00
|
|
|
|
2011-02-27 08:25:45 +00:00
|
|
|
void HighlightWindowEffect::slotWindowClosed(EffectWindow* w)
|
2011-01-30 14:34:42 +00:00
|
|
|
{
|
|
|
|
if (m_monitorWindow == w) // The monitoring window was destroyed
|
2009-02-03 10:51:36 +00:00
|
|
|
finishHighlighting();
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2009-01-31 15:12:14 +00:00
|
|
|
|
2011-02-27 09:47:42 +00:00
|
|
|
void HighlightWindowEffect::slotWindowDeleted(EffectWindow* w)
|
2011-01-30 14:34:42 +00:00
|
|
|
{
|
|
|
|
m_windowOpacity.remove(w);
|
|
|
|
}
|
2009-02-09 12:37:59 +00:00
|
|
|
|
2013-09-03 19:55:39 +00:00
|
|
|
void HighlightWindowEffect::slotPropertyNotify(EffectWindow* w, long a, EffectWindow *addedWindow)
|
2011-01-30 14:34:42 +00:00
|
|
|
{
|
2017-09-10 14:51:18 +00:00
|
|
|
if (a != m_atom || m_atom == XCB_ATOM_NONE)
|
2009-01-31 15:12:14 +00:00
|
|
|
return; // Not our atom
|
|
|
|
|
2010-10-17 19:49:07 +00:00
|
|
|
// if the window is null, the property was set on the root window - see events.cpp
|
2011-01-30 14:34:42 +00:00
|
|
|
QByteArray byteData = w ? w->readProperty(m_atom, m_atom, 32) :
|
|
|
|
effects->readRootProperty(m_atom, m_atom, 32);
|
|
|
|
if (byteData.length() < 1) {
|
|
|
|
// Property was removed, clearing highlight
|
2013-09-03 19:55:39 +00:00
|
|
|
if (!addedWindow || w != addedWindow)
|
|
|
|
finishHighlighting();
|
2009-02-09 12:37:59 +00:00
|
|
|
return;
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2013-09-25 06:54:22 +00:00
|
|
|
auto* data = reinterpret_cast<uint32_t*>(byteData.data());
|
2009-01-31 15:12:14 +00:00
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
if (!data[0]) {
|
|
|
|
// Purposely clearing highlight by issuing a NULL target
|
2009-02-03 10:51:36 +00:00
|
|
|
finishHighlighting();
|
2009-01-31 15:12:14 +00:00
|
|
|
return;
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2009-01-31 15:12:14 +00:00
|
|
|
m_monitorWindow = w;
|
2009-05-05 11:46:36 +00:00
|
|
|
bool found = false;
|
2011-01-30 14:34:42 +00:00
|
|
|
int length = byteData.length() / sizeof(data[0]);
|
|
|
|
//foreach ( EffectWindow* e, m_highlightedWindows )
|
2009-12-20 13:21:00 +00:00
|
|
|
// effects->setElevatedWindow( e, false );
|
2009-09-13 11:36:45 +00:00
|
|
|
m_highlightedWindows.clear();
|
2013-09-03 19:55:39 +00:00
|
|
|
m_highlightedIds.clear();
|
2011-01-30 14:34:42 +00:00
|
|
|
for (int i = 0; i < length; i++) {
|
2013-09-03 19:55:39 +00:00
|
|
|
m_highlightedIds << data[i];
|
2011-01-30 14:34:42 +00:00
|
|
|
EffectWindow* foundWin = effects->findWindow(data[i]);
|
|
|
|
if (!foundWin) {
|
2013-11-29 05:18:28 +00:00
|
|
|
qCDebug(KWINEFFECTS) << "Invalid window targetted for highlight. Requested:" << data[i];
|
2013-09-03 19:55:39 +00:00
|
|
|
continue; // might come in later.
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2011-12-13 13:28:20 +00:00
|
|
|
m_highlightedWindows.append(foundWin);
|
|
|
|
// TODO: We cannot just simply elevate the window as this will elevate it over
|
|
|
|
// Plasma tooltips and other such windows as well
|
|
|
|
//effects->setElevatedWindow( foundWin, true );
|
2011-01-30 14:34:42 +00:00
|
|
|
found = true;
|
|
|
|
}
|
|
|
|
if (!found) {
|
2009-09-13 11:36:45 +00:00
|
|
|
finishHighlighting();
|
2009-05-05 11:46:36 +00:00
|
|
|
return;
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2009-02-03 10:51:36 +00:00
|
|
|
prepareHighlighting();
|
2011-01-30 14:34:42 +00:00
|
|
|
if (w)
|
2010-10-17 19:49:07 +00:00
|
|
|
m_windowOpacity[w] = 1.0; // Because it's not in stackingOrder() yet
|
2009-02-07 15:40:31 +00:00
|
|
|
|
|
|
|
/* TODO: Finish thumbnails of offscreen windows, not sure if it's worth it though
|
2011-01-30 14:34:42 +00:00
|
|
|
if ( !m_highlightedWindow->isOnCurrentDesktop() )
|
2009-02-07 15:40:31 +00:00
|
|
|
{ // Window is offscreen, determine thumbnail position
|
|
|
|
QRect screenArea = effects->clientArea( MaximizeArea ); // Workable area of the active screen
|
|
|
|
QRect outerArea = outerArea.adjusted( outerArea.width() / 10, outerArea.height() / 10,
|
|
|
|
-outerArea.width() / 10, -outerArea.height() / 10 ); // Add 10% margin around the edge
|
|
|
|
QRect innerArea = outerArea.adjusted( outerArea.width() / 40, outerArea.height() / 40,
|
|
|
|
-outerArea.width() / 40, -outerArea.height() / 40 ); // Outer edge of the thumbnail border (2.5%)
|
|
|
|
QRect thumbArea = outerArea.adjusted( 20, 20, -20, -20 ); // Outer edge of the thumbnail (20px)
|
|
|
|
|
|
|
|
// Determine the maximum size that we can make the thumbnail within the innerArea
|
|
|
|
double areaAspect = double( thumbArea.width() ) / double( thumbArea.height() );
|
|
|
|
double windowAspect = aspectRatio( m_highlightedWindow );
|
|
|
|
QRect thumbRect; // Position doesn't matter right now, but it will later
|
2011-01-30 14:34:42 +00:00
|
|
|
if ( windowAspect > areaAspect )
|
2009-02-07 15:40:31 +00:00
|
|
|
// Top/bottom will touch first
|
|
|
|
thumbRect = QRect( 0, 0, widthForHeight( thumbArea.height() ), thumbArea.height() );
|
|
|
|
else // Left/right will touch first
|
|
|
|
thumbRect = QRect( 0, 0, thumbArea.width(), heightForWidth( thumbArea.width() ));
|
2011-01-30 14:34:42 +00:00
|
|
|
if ( thumbRect.width() >= m_highlightedWindow->width() )
|
2009-02-07 15:40:31 +00:00
|
|
|
// Area is larger than the window, just use the window's size
|
|
|
|
thumbRect = m_highlightedWindow->geometry();
|
|
|
|
|
|
|
|
// Determine position of desktop relative to the current one
|
2009-02-14 14:49:46 +00:00
|
|
|
QPoint direction = effects->desktopGridCoords( m_highlightedWindow->desktop() ) -
|
|
|
|
effects->desktopGridCoords( effects->currentDesktop() );
|
2009-02-07 15:40:31 +00:00
|
|
|
|
|
|
|
// Draw a line from the center of the current desktop to the center of the target desktop.
|
|
|
|
QPointF desktopLine( 0, 0, direction.x() * screenArea.width(), direction.y() * screenArea.height() );
|
|
|
|
desktopLeft.translate( screenArea.width() / 2, screenArea.height() / 2 ); // Move to the screen center
|
|
|
|
|
|
|
|
// Take the point where the line crosses the outerArea, this will be the tip of our arrow
|
|
|
|
QPointF arrowTip;
|
|
|
|
QLineF testLine( // Top
|
|
|
|
outerArea.x(), outerArea.y(),
|
|
|
|
outerArea.x() + outerArea.width(), outerArea.y() );
|
2011-01-30 14:34:42 +00:00
|
|
|
if ( desktopLine.intersect( testLine, &arrowTip ) != QLineF::BoundedIntersection )
|
2009-02-07 15:40:31 +00:00
|
|
|
{
|
|
|
|
testLine = QLineF( // Right
|
|
|
|
outerArea.x() + outerArea.width(), outerArea.y(),
|
|
|
|
outerArea.x() + outerArea.width(), outerArea.y() + outerArea.height() );
|
2011-01-30 14:34:42 +00:00
|
|
|
if ( desktopLine.intersect( testLine, &arrowTip ) != QLineF::BoundedIntersection )
|
2009-02-07 15:40:31 +00:00
|
|
|
{
|
|
|
|
testLine = QLineF( // Bottom
|
|
|
|
outerArea.x() + outerArea.width(), outerArea.y() + outerArea.height(),
|
|
|
|
outerArea.x(), outerArea.y() + outerArea.height() );
|
2011-01-30 14:34:42 +00:00
|
|
|
if ( desktopLine.intersect( testLine, &arrowTip ) != QLineF::BoundedIntersection )
|
2009-02-07 15:40:31 +00:00
|
|
|
{
|
|
|
|
testLine = QLineF( // Left
|
|
|
|
outerArea.x(), outerArea.y() + outerArea.height(),
|
|
|
|
outerArea.x(), outerArea.y() );
|
|
|
|
desktopLine.intersect( testLine, &arrowTip ); // Should never fail
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
m_arrowTip = arrowTip.toPoint();
|
|
|
|
} */
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2009-02-03 10:51:36 +00:00
|
|
|
|
|
|
|
void HighlightWindowEffect::prepareHighlighting()
|
2011-01-30 14:34:42 +00:00
|
|
|
{
|
2009-02-03 10:51:36 +00:00
|
|
|
// Create window data for every window. Just calling [w] creates it.
|
|
|
|
m_finishing = false;
|
2011-12-13 13:28:20 +00:00
|
|
|
foreach (EffectWindow * w, effects->stackingOrder()) {
|
|
|
|
if (!m_windowOpacity.contains(w)) // Just in case we are still finishing from last time
|
|
|
|
m_windowOpacity.insertMulti(w, isInitiallyHidden(w) ? 0.0 : 1.0);
|
|
|
|
if (!m_highlightedWindows.isEmpty())
|
|
|
|
m_highlightedWindows.at(0)->addRepaintFull();
|
|
|
|
}
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2009-02-03 10:51:36 +00:00
|
|
|
|
|
|
|
void HighlightWindowEffect::finishHighlighting()
|
2011-01-30 14:34:42 +00:00
|
|
|
{
|
2009-02-03 10:51:36 +00:00
|
|
|
m_finishing = true;
|
|
|
|
m_monitorWindow = NULL;
|
2009-05-05 11:46:36 +00:00
|
|
|
m_highlightedWindows.clear();
|
2011-02-06 15:50:47 +00:00
|
|
|
if (!m_windowOpacity.isEmpty())
|
|
|
|
m_windowOpacity.constBegin().key()->addRepaintFull();
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2009-01-31 15:12:14 +00:00
|
|
|
|
2016-08-30 13:50:31 +00:00
|
|
|
void HighlightWindowEffect::highlightWindows(const QVector<KWin::EffectWindow *> &windows)
|
|
|
|
{
|
|
|
|
if (windows.isEmpty()) {
|
|
|
|
finishHighlighting();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
m_monitorWindow = nullptr;
|
|
|
|
m_highlightedWindows.clear();
|
|
|
|
m_highlightedIds.clear();
|
|
|
|
for (auto w : windows) {
|
|
|
|
m_highlightedWindows << w;
|
|
|
|
}
|
|
|
|
prepareHighlighting();
|
|
|
|
}
|
|
|
|
|
2011-08-27 09:21:31 +00:00
|
|
|
bool HighlightWindowEffect::isActive() const
|
|
|
|
{
|
2014-02-12 17:16:52 +00:00
|
|
|
return !(m_windowOpacity.isEmpty() || effects->isScreenLocked());
|
2011-08-27 09:21:31 +00:00
|
|
|
}
|
|
|
|
|
2016-08-30 13:50:31 +00:00
|
|
|
bool HighlightWindowEffect::provides(Feature feature)
|
|
|
|
{
|
|
|
|
switch (feature) {
|
|
|
|
case HighlightWindows:
|
|
|
|
return true;
|
|
|
|
default:
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
bool HighlightWindowEffect::perform(Feature feature, const QVariantList &arguments)
|
|
|
|
{
|
|
|
|
if (feature != HighlightWindows) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (arguments.size() != 1) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
highlightWindows(arguments.first().value<QVector<EffectWindow*>>());
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2009-01-31 15:12:14 +00:00
|
|
|
} // namespace
|