You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
645 lines
26 KiB
645 lines
26 KiB
/* |
|
KWin - the KDE window manager |
|
This file is part of the KDE project. |
|
|
|
SPDX-FileCopyrightText: 1999, 2000 Matthias Ettrich <ettrich@kde.org> |
|
SPDX-FileCopyrightText: 2003 Lubos Lunak <l.lunak@kde.org> |
|
|
|
SPDX-License-Identifier: GPL-2.0-or-later |
|
*/ |
|
|
|
/* |
|
|
|
This file contains things relevant to window activation and focus |
|
stealing prevention. |
|
|
|
*/ |
|
|
|
#include "cursor.h" |
|
#include "focuschain.h" |
|
#include "workspace.h" |
|
#if KWIN_BUILD_ACTIVITIES |
|
#include "activities.h" |
|
#endif |
|
#include "rules.h" |
|
#include "useractions.h" |
|
#include "virtualdesktops.h" |
|
#include "waylandwindow.h" |
|
#include "window.h" |
|
|
|
#if KWIN_BUILD_X11 |
|
#include "atoms.h" |
|
#include "group.h" |
|
#include "netinfo.h" |
|
#include "x11window.h" |
|
#include <kstartupinfo.h> |
|
#endif |
|
|
|
#include <KLocalizedString> |
|
#include <QDebug> |
|
#include <kstringhandler.h> |
|
|
|
namespace KWin |
|
{ |
|
|
|
/* |
|
Prevention of focus stealing: |
|
|
|
KWin tries to prevent unwanted changes of focus, that would result |
|
from mapping a new window. Also, some nasty applications may try |
|
to force focus change even in cases when ICCCM 4.2.7 doesn't allow it |
|
(e.g. they may try to activate their main window because the user |
|
definitely "needs" to see something happened - misusing |
|
of QWidget::setActiveWindow() may be such case). |
|
|
|
There are 4 ways how a window may become active: |
|
- the user changes the active window (e.g. focus follows mouse, clicking |
|
on some window's titlebar) - the change of focus will |
|
be done by KWin, so there's nothing to solve in this case |
|
- the change of active window will be requested using the _NET_ACTIVE_WINDOW |
|
message (handled in RootInfo::changeActiveWindow()) - such requests |
|
will be obeyed, because this request is meant mainly for e.g. taskbar |
|
asking the WM to change the active window as a result of some user action. |
|
Normal applications should use this request only rarely in special cases. |
|
See also below the discussion of _NET_ACTIVE_WINDOW_TRANSFER. |
|
- the change of active window will be done by performing XSetInputFocus() |
|
on a window that's not currently active. ICCCM 4.2.7 describes when |
|
the application may perform change of input focus. In order to handle |
|
misbehaving applications, KWin will try to detect focus changes to |
|
windows that don't belong to currently active application, and restore |
|
focus back to the currently active window, instead of activating the window |
|
that got focus (unfortunately there's no way to FocusChangeRedirect similar |
|
to e.g. SubstructureRedirect, so there will be short time when the focus |
|
will be changed). The check itself that's done is |
|
Workspace::allowWindowActivation() (see below). |
|
- a new window will be mapped - this is the most complicated case. If |
|
the new window belongs to the currently active application, it may be safely |
|
mapped on top and activated. The same if there's no active window, |
|
or the active window is the desktop. These checks are done by |
|
Workspace::allowWindowActivation(). |
|
Following checks need to compare times. One time is the timestamp |
|
of last user action in the currently active window, the other time is |
|
the timestamp of the action that originally caused mapping of the new window |
|
(e.g. when the application was started). If the first time is newer than |
|
the second one, the window will not be activated, as that indicates |
|
futher user actions took place after the action leading to this new |
|
mapped window. This check is done by Workspace::allowWindowActivation(). |
|
There are several ways how to get the timestamp of action that caused |
|
the new mapped window (done in X11Window::readUserTimeMapTimestamp()) : |
|
- the window may have the _NET_WM_USER_TIME property. This way |
|
the application may either explicitly request that the window is not |
|
activated (by using 0 timestamp), or the property contains the time |
|
of last user action in the application. |
|
- KWin itself tries to detect time of last user action in every window, |
|
by watching KeyPress and ButtonPress events on windows. This way some |
|
events may be missed (if they don't propagate to the toplevel window), |
|
but it's good as a fallback for applications that don't provide |
|
_NET_WM_USER_TIME, and missing some events may at most lead |
|
to unwanted focus stealing. |
|
- the timestamp may come from application startup notification. |
|
Application startup notification, if it exists for the new mapped window, |
|
should include time of the user action that caused it. |
|
- if there's no timestamp available, it's checked whether the new window |
|
belongs to some already running application - if yes, the timestamp |
|
will be 0 (i.e. refuse activation) |
|
- if the window is from session restored window, the timestamp will |
|
be 0 too, unless this application was the active one at the time |
|
when the session was saved, in which case the window will be |
|
activated if there wasn't any user interaction since the time |
|
KWin was started. |
|
- as the last resort, the _KDE_NET_USER_CREATION_TIME timestamp |
|
is used. For every toplevel window that is created (see CreateNotify |
|
handling), this property is set to the at that time current time. |
|
Since at this time it's known that the new window doesn't belong |
|
to any existing application (better said, the application doesn't |
|
have any other window mapped), it is either the very first window |
|
of the application, or it is the only window of the application |
|
that was hidden before. The latter case is handled by removing |
|
the property from windows before withdrawing them, making |
|
the timestamp empty for next mapping of the window. In the sooner |
|
case, the timestamp will be used. This helps in case when |
|
an application is launched without application startup notification, |
|
it creates its mainwindow, and starts its initialization (that |
|
may possibly take long time). The timestamp used will be older |
|
than any user action done after launching this application. |
|
- if no timestamp is found at all, the window is activated. |
|
The check whether two windows belong to the same application (same |
|
process) is done in X11Window::belongToSameApplication(). Not 100% reliable, |
|
but hopefully 99,99% reliable. |
|
|
|
As a somewhat special case, window activation is always enabled when |
|
session saving is in progress. When session saving, the session |
|
manager allows only one application to interact with the user. |
|
Not allowing window activation in such case would result in e.g. dialogs |
|
not becoming active, so focus stealing prevention would cause here |
|
more harm than good. |
|
|
|
Windows that attempted to become active but KWin prevented this will |
|
be marked as demanding user attention. They'll get |
|
the _NET_WM_STATE_DEMANDS_ATTENTION state, and the taskbar should mark |
|
them specially (blink, etc.). The state will be reset when the window |
|
eventually really becomes active. |
|
|
|
There are two more ways how a window can become obtrusive, window stealing |
|
focus: By showing above the active window, by either raising itself, |
|
or by moving itself on the active desktop. |
|
- KWin will refuse raising non-active window above the active one, |
|
unless they belong to the same application. Applications shouldn't |
|
raise their windows anyway (unless the app wants to raise one |
|
of its windows above another of its windows). |
|
- KWin activates windows moved to the current desktop (as that seems |
|
logical from the user's point of view, after sending the window |
|
there directly from KWin, or e.g. using pager). This means |
|
applications shouldn't send their windows to another desktop |
|
(SELI TODO - but what if they do?) |
|
|
|
Special cases I can think of: |
|
- konqueror reusing, i.e. kfmclient tells running Konqueror instance |
|
to open new window |
|
- without focus stealing prevention - no problem |
|
- with ASN (application startup notification) - ASN is forwarded, |
|
and because it's newer than the instance's user timestamp, |
|
it takes precedence |
|
- without ASN - user timestamp needs to be reset, otherwise it would |
|
be used, and it's old; moreover this new window mustn't be detected |
|
as window belonging to already running application, or it wouldn't |
|
be activated - see X11Window::sameAppWindowRoleMatch() for the (rather ugly) |
|
hack |
|
- konqueror preloading, i.e. window is created in advance, and kfmclient |
|
tells this Konqueror instance to show it later |
|
- without focus stealing prevention - no problem |
|
- with ASN - ASN is forwarded, and because it's newer than the instance's |
|
user timestamp, it takes precedence |
|
- without ASN - user timestamp needs to be reset, otherwise it would |
|
be used, and it's old; also, creation timestamp is changed to |
|
the time the instance starts (re-)initializing the window, |
|
this ensures creation timestamp will still work somewhat even in this case |
|
- KUniqueApplication - when the window is already visible, and the new instance |
|
wants it to activate |
|
- without focus stealing prevention - _NET_ACTIVE_WINDOW - no problem |
|
- with ASN - ASN is forwarded, and set on the already visible window, KWin |
|
treats the window as new with that ASN |
|
- without ASN - _NET_ACTIVE_WINDOW as application request is used, |
|
and there's no really usable timestamp, only timestamp |
|
from the time the (new) application instance was started, |
|
so KWin will activate the window *sigh* |
|
- the bad thing here is that there's absolutely no chance to recognize |
|
the case of starting this KUniqueApp from Konsole (and thus wanting |
|
the already visible window to become active) from the case |
|
when something started this KUniqueApp without ASN (in which case |
|
the already visible window shouldn't become active) |
|
- the only solution is using ASN for starting applications, at least silent |
|
(i.e. without feedback) |
|
- when one application wants to activate another application's window (e.g. KMail |
|
activating already running KAddressBook window ?) |
|
- without focus stealing prevention - _NET_ACTIVE_WINDOW - no problem |
|
- with ASN - can't be here, it's the KUniqueApp case then |
|
- without ASN - _NET_ACTIVE_WINDOW as application request should be used, |
|
KWin will activate the new window depending on the timestamp and |
|
whether it belongs to the currently active application |
|
|
|
_NET_ACTIVE_WINDOW usage: |
|
data.l[0]= 1 ->app request |
|
= 2 ->pager request |
|
= 0 - backwards compatibility |
|
data.l[1]= timestamp |
|
*/ |
|
|
|
//**************************************** |
|
// Workspace |
|
//**************************************** |
|
|
|
/** |
|
* Informs the workspace about the active window, i.e. the window that |
|
* has the focus (or None if no window has the focus). This functions |
|
* is called by the window itself that gets focus. It has no other |
|
* effect than fixing the focus chain and the return value of |
|
* activeWindow(). And of course, to propagate the active window to the |
|
* world. |
|
*/ |
|
void Workspace::setActiveWindow(Window *window) |
|
{ |
|
if (m_activeWindow == window) { |
|
return; |
|
} |
|
|
|
if (active_popup && m_activePopupWindow != window && m_setActiveWindowRecursion == 0) { |
|
closeActivePopup(); |
|
} |
|
if (m_userActionsMenu->hasWindow() && !m_userActionsMenu->isMenuWindow(window) && m_setActiveWindowRecursion == 0) { |
|
m_userActionsMenu->close(); |
|
} |
|
StackingUpdatesBlocker blocker(this); |
|
++m_setActiveWindowRecursion; |
|
updateFocusMousePosition(Cursors::self()->mouse()->pos()); |
|
|
|
if (qobject_cast<WaylandWindow *>(window)) { |
|
// focusIn events only arrive for X11 windows, Wayland windows don't use such a mechanism |
|
// and so X11 windows could wrongly get stuck in the list |
|
should_get_focus.clear(); |
|
} |
|
|
|
if (m_activeWindow != nullptr) { |
|
// note that this may call setActiveWindow( NULL ), therefore the recursion counter |
|
m_activeWindow->setActive(false); |
|
} |
|
m_activeWindow = window; |
|
Q_ASSERT(window == nullptr || window->isActive()); |
|
|
|
if (m_activeWindow) { |
|
m_lastActiveWindow = m_activeWindow; |
|
m_focusChain->update(m_activeWindow, FocusChain::MakeFirst); |
|
m_activeWindow->demandAttention(false); |
|
|
|
// activating a client can cause a non active fullscreen window to loose the ActiveLayer status on > 1 screens |
|
if (outputs().count() > 1) { |
|
for (auto it = m_windows.begin(); it != m_windows.end(); ++it) { |
|
if (*it != m_activeWindow && (*it)->layer() == ActiveLayer && (*it)->output() == m_activeWindow->output()) { |
|
(*it)->updateLayer(); |
|
} |
|
} |
|
} |
|
} |
|
|
|
if (window) { |
|
disableGlobalShortcutsForClient(window->rules()->checkDisableGlobalShortcuts(false)); |
|
} else { |
|
disableGlobalShortcutsForClient(false); |
|
} |
|
|
|
updateStackingOrder(); // e.g. fullscreens have different layer when active/not-active |
|
|
|
#if KWIN_BUILD_X11 |
|
if (rootInfo()) { |
|
rootInfo()->setActiveClient(m_activeWindow); |
|
} |
|
#endif |
|
|
|
Q_EMIT windowActivated(m_activeWindow); |
|
--m_setActiveWindowRecursion; |
|
} |
|
|
|
/** |
|
* Tries to activate the window \a window. This function performs what you |
|
* expect when clicking the respective entry in a taskbar: showing and |
|
* raising the window (this may imply switching to the another virtual |
|
* desktop) and putting the focus onto it. Once X really gave focus to |
|
* the window window as requested, the window itself will call |
|
* setActiveWindow() and the operation is complete. This may not happen |
|
* with certain focus policies, though. |
|
* |
|
* @see setActiveWindow |
|
* @see requestFocus |
|
*/ |
|
void Workspace::activateWindow(Window *window, bool force) |
|
{ |
|
if (window == nullptr) { |
|
focusToNull(); |
|
setActiveWindow(nullptr); |
|
return; |
|
} |
|
if (!window->isClient() || window->isDeleted() || !window->wantsInput()) { |
|
return; |
|
} |
|
if (window->isHiddenByShowDesktop()) { |
|
++block_focus; |
|
setShowingDesktop(false); |
|
--block_focus; |
|
} |
|
raiseWindow(window); |
|
if (!window->isOnCurrentDesktop()) { |
|
++block_focus; |
|
switch (options->activationDesktopPolicy()) { |
|
case Options::ActivationDesktopPolicy::SwitchToOtherDesktop: |
|
VirtualDesktopManager::self()->setCurrent(window->desktops().constLast()); |
|
break; |
|
case Options::ActivationDesktopPolicy::BringToCurrentDesktop: |
|
window->enterDesktop(VirtualDesktopManager::self()->currentDesktop()); |
|
break; |
|
case Options::ActivationDesktopPolicy::DoNothing: |
|
break; |
|
} |
|
--block_focus; |
|
} |
|
#if KWIN_BUILD_ACTIVITIES |
|
if (!window->isOnCurrentActivity()) { |
|
++block_focus; |
|
// DBUS! |
|
// first isn't necessarily best, but it's easiest |
|
m_activities->setCurrent(window->activities().constFirst()); |
|
--block_focus; |
|
} |
|
#endif |
|
if (window->isMinimized()) { |
|
window->setMinimized(false); |
|
} |
|
|
|
// ensure the window is really visible - could eg. be a hidden utility window, see bug #348083 |
|
window->setHidden(false); |
|
|
|
// TODO force should perhaps allow this only if the window already contains the mouse |
|
if (options->focusPolicyIsReasonable() || force) { |
|
requestFocus(window, force); |
|
} |
|
|
|
#if KWIN_BUILD_X11 |
|
// Don't update user time for windows that have focus stealing workaround. |
|
// As they usually belong to the current active window but fail to provide |
|
// this information, updating their user time would make the user time |
|
// of the currently active window old, and reject further activation for it. |
|
// E.g. typing URL in minicli which will show kio_uiserver dialog (with workaround), |
|
// and then kdesktop shows dialog about SSL certificate. |
|
// This needs also avoiding user creation time in X11Window::readUserTimeMapTimestamp(). |
|
if (X11Window *x11Window = dynamic_cast<X11Window *>(window)) { |
|
// updateUserTime is X11 specific |
|
x11Window->updateUserTime(); |
|
} |
|
#endif |
|
} |
|
|
|
/** |
|
* Tries to activate the window by asking X for the input focus. This |
|
* function does not perform any show, raise or desktop switching. See |
|
* Workspace::activateWindow() instead. |
|
* |
|
* @see activateWindow |
|
*/ |
|
bool Workspace::requestFocus(Window *window, bool force) |
|
{ |
|
return takeActivity(window, force ? ActivityFocusForce : ActivityFocus); |
|
} |
|
|
|
bool Workspace::takeActivity(Window *window, ActivityFlags flags) |
|
{ |
|
// the 'if ( window == m_activeWindow ) return;' optimization mustn't be done here |
|
if (!focusChangeEnabled() && (window != m_activeWindow)) { |
|
flags &= ~ActivityFocus; |
|
} |
|
|
|
if (!window) { |
|
focusToNull(); |
|
return true; |
|
} |
|
|
|
if (flags & ActivityFocus) { |
|
Window *modal = window->findModal(); |
|
if (modal != nullptr && modal != window) { |
|
if (modal->desktops() != window->desktops()) { |
|
modal->setDesktops(window->desktops()); |
|
} |
|
if (!modal->isShown() && !modal->isMinimized()) { // forced desktop or utility window |
|
activateWindow(modal); // activating a minimized blocked window will unminimize its modal implicitly |
|
} |
|
// if the click was inside the window (i.e. handled is set), |
|
// but it has a modal, there's no need to use handled mode, because |
|
// the modal doesn't get the click anyway |
|
// raising of the original window needs to be still done |
|
if (flags & ActivityRaise) { |
|
raiseWindow(window); |
|
} |
|
window = modal; |
|
} |
|
cancelDelayFocus(); |
|
} |
|
if (!flags.testFlag(ActivityFocusForce) && (window->isDock() || window->isSplash())) { |
|
// toplevel menus and dock windows don't take focus if not forced |
|
// and don't have a flag that they take focus |
|
if (!window->dockWantsInput()) { |
|
flags &= ~ActivityFocus; |
|
} |
|
} |
|
if (window->isShade()) { |
|
if (window->wantsInput() && (flags & ActivityFocus)) { |
|
// window cannot accept focus, but at least the window should be active (window menu, et. al. ) |
|
window->setActive(true); |
|
focusToNull(); |
|
} |
|
flags &= ~ActivityFocus; |
|
} |
|
if (!window->isShown()) { // shouldn't happen, call activateWindow() if needed |
|
qCWarning(KWIN_CORE) << "takeActivity: not shown"; |
|
return false; |
|
} |
|
|
|
bool ret = true; |
|
|
|
if (flags & ActivityFocus) { |
|
ret &= window->takeFocus(); |
|
} |
|
if (flags & ActivityRaise) { |
|
workspace()->raiseWindow(window); |
|
} |
|
|
|
if (!window->isOnActiveOutput()) { |
|
setActiveOutput(window->output()); |
|
} |
|
|
|
return ret; |
|
} |
|
|
|
Window *Workspace::windowUnderMouse(Output *output) const |
|
{ |
|
auto it = stackingOrder().constEnd(); |
|
while (it != stackingOrder().constBegin()) { |
|
auto window = *(--it); |
|
if (!window->isClient()) { |
|
continue; |
|
} |
|
|
|
// rule out windows which are not really visible. |
|
// the screen test is rather superfluous for xrandr & twinview since the geometry would differ -> TODO: might be dropped |
|
if (!(window->isShown() && window->isOnCurrentDesktop() && window->isOnCurrentActivity() && window->isOnOutput(output) && !window->isShade())) { |
|
continue; |
|
} |
|
|
|
if (exclusiveContains(window->frameGeometry(), Cursors::self()->mouse()->pos())) { |
|
return window; |
|
} |
|
} |
|
return nullptr; |
|
} |
|
|
|
// deactivates 'window' and activates next window |
|
bool Workspace::activateNextWindow(Window *window) |
|
{ |
|
// if 'c' is not the active or the to-become active one, do nothing |
|
if (!(window == m_activeWindow || (should_get_focus.count() > 0 && window == should_get_focus.last()))) { |
|
return false; |
|
} |
|
|
|
closeActivePopup(); |
|
|
|
if (window != nullptr) { |
|
if (window == m_activeWindow) { |
|
setActiveWindow(nullptr); |
|
} |
|
should_get_focus.removeAll(window); |
|
} |
|
|
|
// if blocking focus, move focus to the desktop later if needed |
|
// in order to avoid flickering |
|
if (!focusChangeEnabled()) { |
|
focusToNull(); |
|
return true; |
|
} |
|
|
|
if (!options->focusPolicyIsReasonable()) { |
|
return false; |
|
} |
|
|
|
Window *focusCandidate = nullptr; |
|
|
|
VirtualDesktop *desktop = VirtualDesktopManager::self()->currentDesktop(); |
|
Output *output = window ? window->output() : workspace()->activeOutput(); |
|
|
|
if (!focusCandidate && showingDesktop()) { |
|
focusCandidate = findDesktop(desktop, output); // to not break the state |
|
} |
|
|
|
if (!focusCandidate && options->isNextFocusPrefersMouse()) { |
|
focusCandidate = windowUnderMouse(output); |
|
if (focusCandidate && (focusCandidate == window || focusCandidate->isDesktop())) { |
|
// should rather not happen, but it cannot get the focus. rest of usability is tested above |
|
focusCandidate = nullptr; |
|
} |
|
} |
|
|
|
if (!focusCandidate) { // no suitable window under the mouse -> find sth. else |
|
// first try to pass the focus to the (former) active clients leader |
|
if (window && window->isTransient()) { |
|
auto leaders = window->mainWindows(); |
|
if (leaders.count() == 1 && m_focusChain->isUsableFocusCandidate(leaders.at(0), window)) { |
|
focusCandidate = leaders.at(0); |
|
raiseWindow(focusCandidate); // also raise - we don't know where it came from |
|
} |
|
} |
|
if (!focusCandidate) { |
|
// nope, ask the focus chain for the next candidate |
|
focusCandidate = m_focusChain->nextForDesktop(window, desktop); |
|
} |
|
} |
|
|
|
if (focusCandidate == nullptr) { // last chance: focus the desktop |
|
focusCandidate = findDesktop(desktop, output); |
|
} |
|
|
|
if (focusCandidate != nullptr) { |
|
requestFocus(focusCandidate); |
|
} else { |
|
focusToNull(); |
|
} |
|
|
|
return true; |
|
} |
|
|
|
void Workspace::switchToOutput(Output *output) |
|
{ |
|
if (!options->focusPolicyIsReasonable()) { |
|
return; |
|
} |
|
closeActivePopup(); |
|
VirtualDesktop *desktop = VirtualDesktopManager::self()->currentDesktop(); |
|
Window *get_focus = m_focusChain->getForActivation(desktop, output); |
|
if (get_focus == nullptr) { |
|
get_focus = findDesktop(desktop, output); |
|
} |
|
if (get_focus != nullptr && get_focus != mostRecentlyActivatedWindow()) { |
|
requestFocus(get_focus); |
|
} |
|
setActiveOutput(output); |
|
} |
|
|
|
void Workspace::gotFocusIn(const Window *window) |
|
{ |
|
if (should_get_focus.contains(window)) { |
|
// remove also all sooner elements that should have got FocusIn, |
|
// but didn't for some reason (and also won't anymore, because they were sooner) |
|
while (should_get_focus.first() != window) { |
|
should_get_focus.pop_front(); |
|
} |
|
should_get_focus.pop_front(); // remove 'window' |
|
} |
|
} |
|
|
|
void Workspace::setShouldGetFocus(Window *window) |
|
{ |
|
should_get_focus.append(window); |
|
updateStackingOrder(); // e.g. fullscreens have different layer when active/not-active |
|
} |
|
|
|
// basically the same like allowWindowActivation(), this time allowing |
|
// a window to be fully raised upon its own request (XRaiseWindow), |
|
// if refused, it will be raised only on top of windows belonging |
|
// to the same application |
|
bool Workspace::allowFullClientRaising(const KWin::Window *window, uint32_t time) |
|
{ |
|
int level = window->rules()->checkFSP(options->focusStealingPreventionLevel()); |
|
if (sessionManager()->state() == SessionState::Saving && level <= 2) { // <= normal |
|
return true; |
|
} |
|
Window *ac = mostRecentlyActivatedWindow(); |
|
if (level == 0) { // none |
|
return true; |
|
} |
|
if (level == 4) { // extreme |
|
return false; |
|
} |
|
if (ac == nullptr || ac->isDesktop()) { |
|
qCDebug(KWIN_CORE) << "Raising: No window active, allowing"; |
|
return true; // no active window -> always allow |
|
} |
|
// TODO window urgency -> return true? |
|
if (Window::belongToSameApplication(window, ac, Window::SameApplicationCheck::RelaxedForActive)) { |
|
qCDebug(KWIN_CORE) << "Raising: Belongs to active application"; |
|
return true; |
|
} |
|
if (level == 3) { // high |
|
return false; |
|
} |
|
#if KWIN_BUILD_X11 |
|
xcb_timestamp_t user_time = ac->userTime(); |
|
qCDebug(KWIN_CORE) << "Raising, compared:" << time << ":" << user_time |
|
<< ":" << (NET::timestampCompare(time, user_time) >= 0); |
|
return NET::timestampCompare(time, user_time) >= 0; // time >= user_time |
|
#else |
|
return true; |
|
#endif |
|
} |
|
|
|
/** |
|
* Called from X11Window after FocusIn that wasn't initiated by KWin and the window wasn't |
|
* allowed to activate. |
|
* |
|
* Returns @c true if the focus has been restored successfully; otherwise returns @c false. |
|
*/ |
|
bool Workspace::restoreFocus() |
|
{ |
|
// this updateXTime() is necessary - as FocusIn events don't have |
|
// a timestamp *sigh*, kwin's timestamp would be older than the timestamp |
|
// that was used by whoever caused the focus change, and therefore |
|
// the attempt to restore the focus would fail due to old timestamp |
|
#if KWIN_BUILD_X11 |
|
kwinApp()->updateXTime(); |
|
#endif |
|
if (should_get_focus.count() > 0) { |
|
return requestFocus(should_get_focus.last()); |
|
} else if (m_lastActiveWindow) { |
|
return requestFocus(m_lastActiveWindow); |
|
} |
|
return true; |
|
} |
|
|
|
void Workspace::windowAttentionChanged(Window *window, bool set) |
|
{ |
|
if (window->isDeleted()) { |
|
return; |
|
} |
|
if (set) { |
|
attention_chain.removeAll(window); |
|
attention_chain.prepend(window); |
|
} else { |
|
attention_chain.removeAll(window); |
|
} |
|
} |
|
|
|
} // namespace
|
|
|