Ensure that the start Client to build up the ClientModel is in the Focus Chain
If the start Client is not part of the focus chain the call to nextClientFocusChain() cannot return the Client again. So the loop break condition is never reached and as the focus chain is not empty the call always returns a not null Client which means KWin is caught in an endless loop. This change checks that the starting Client is in the focus chain and if not the first Client of the focus chain is used. BUG: 306260 BUG: 306275 FIXED-IN: 4.9.2
This commit is contained in:
parent
b92557a0af
commit
ed3effa2d3
8 changed files with 69 additions and 1 deletions
|
@ -188,7 +188,7 @@ void ClientModel::createClientList(int desktop, bool partialReset)
|
||||||
switch(tabBox->config().clientSwitchingMode()) {
|
switch(tabBox->config().clientSwitchingMode()) {
|
||||||
case TabBoxConfig::FocusChainSwitching: {
|
case TabBoxConfig::FocusChainSwitching: {
|
||||||
TabBoxClient* c = start;
|
TabBoxClient* c = start;
|
||||||
if (!c) {
|
if (!tabBox->isInFocusChain(c)) {
|
||||||
QSharedPointer<TabBoxClient> firstClient = tabBox->firstClientFocusChain().toStrongRef();
|
QSharedPointer<TabBoxClient> firstClient = tabBox->firstClientFocusChain().toStrongRef();
|
||||||
if (firstClient) {
|
if (firstClient) {
|
||||||
c = firstClient.data();
|
c = firstClient.data();
|
||||||
|
|
|
@ -112,6 +112,14 @@ QWeakPointer< TabBoxClient > TabBoxHandlerImpl::firstClientFocusChain() const
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool TabBoxHandlerImpl::isInFocusChain(TabBoxClient *client) const
|
||||||
|
{
|
||||||
|
if (TabBoxClientImpl *c = static_cast<TabBoxClientImpl*>(client)) {
|
||||||
|
return Workspace::self()->globalFocusChain().contains(c->client());
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
int TabBoxHandlerImpl::nextDesktopFocusChain(int desktop) const
|
int TabBoxHandlerImpl::nextDesktopFocusChain(int desktop) const
|
||||||
{
|
{
|
||||||
return m_tabBox->nextDesktopFocusChain(desktop);
|
return m_tabBox->nextDesktopFocusChain(desktop);
|
||||||
|
|
|
@ -52,6 +52,7 @@ public:
|
||||||
virtual QString desktopName(int desktop) const;
|
virtual QString desktopName(int desktop) const;
|
||||||
virtual QWeakPointer< TabBoxClient > nextClientFocusChain(TabBoxClient* client) const;
|
virtual QWeakPointer< TabBoxClient > nextClientFocusChain(TabBoxClient* client) const;
|
||||||
virtual QWeakPointer< TabBoxClient > firstClientFocusChain() const;
|
virtual QWeakPointer< TabBoxClient > firstClientFocusChain() const;
|
||||||
|
virtual bool isInFocusChain (TabBoxClient* client) const;
|
||||||
virtual int nextDesktopFocusChain(int desktop) const;
|
virtual int nextDesktopFocusChain(int desktop) const;
|
||||||
virtual int numberOfDesktops() const;
|
virtual int numberOfDesktops() const;
|
||||||
virtual TabBoxClientList stackingOrder() const;
|
virtual TabBoxClientList stackingOrder() const;
|
||||||
|
|
|
@ -119,6 +119,20 @@ public:
|
||||||
* @since 4.9.1
|
* @since 4.9.1
|
||||||
**/
|
**/
|
||||||
virtual QWeakPointer<TabBoxClient> firstClientFocusChain() const = 0;
|
virtual QWeakPointer<TabBoxClient> firstClientFocusChain() const = 0;
|
||||||
|
/**
|
||||||
|
* Checks whether the given @p client is part of the focus chain at all.
|
||||||
|
* This is useful to figure out whether the currently active Client can be used
|
||||||
|
* as a starting point to construct the recently used list.
|
||||||
|
*
|
||||||
|
* In case the @p client is not in the focus chain it is recommended to use the
|
||||||
|
* Client returned by @link firstClientFocusChain.
|
||||||
|
*
|
||||||
|
* The method accepts a @c null Client and in that case @c false is returned.
|
||||||
|
* @param client The Client to check whether it is in the Focus Chain
|
||||||
|
* @return @c true in case the Client is part of the focus chain, @c false otherwise.
|
||||||
|
* @since 4.9.2
|
||||||
|
**/
|
||||||
|
virtual bool isInFocusChain(TabBoxClient* client) const = 0;
|
||||||
/**
|
/**
|
||||||
* @param client The client whose desktop name should be retrieved
|
* @param client The client whose desktop name should be retrieved
|
||||||
* @return The desktop name of the given TabBoxClient. If the client is
|
* @return The desktop name of the given TabBoxClient. If the client is
|
||||||
|
|
|
@ -72,6 +72,9 @@ QWeakPointer< TabBox::TabBoxClient > MockTabBoxHandler::nextClientFocusChain(Tab
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
if (!m_windows.isEmpty()) {
|
||||||
|
return QWeakPointer< TabBox::TabBoxClient >(m_windows.last());
|
||||||
|
}
|
||||||
return QWeakPointer< TabBox::TabBoxClient >();
|
return QWeakPointer< TabBox::TabBoxClient >();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -83,6 +86,20 @@ QWeakPointer< TabBox::TabBoxClient > MockTabBoxHandler::firstClientFocusChain()
|
||||||
return m_windows.first();
|
return m_windows.first();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool MockTabBoxHandler::isInFocusChain(TabBox::TabBoxClient *client) const
|
||||||
|
{
|
||||||
|
if (!client) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
QList< QSharedPointer< TabBox::TabBoxClient > >::const_iterator it = m_windows.constBegin();
|
||||||
|
for (; it != m_windows.constEnd(); ++it) {
|
||||||
|
if ((*it).data() == client) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
QWeakPointer< TabBox::TabBoxClient > MockTabBoxHandler::createMockWindow(const QString &caption, WId id)
|
QWeakPointer< TabBox::TabBoxClient > MockTabBoxHandler::createMockWindow(const QString &caption, WId id)
|
||||||
{
|
{
|
||||||
QSharedPointer< TabBox::TabBoxClient > client(new MockTabBoxClient(caption, id));
|
QSharedPointer< TabBox::TabBoxClient > client(new MockTabBoxClient(caption, id));
|
||||||
|
|
|
@ -60,6 +60,7 @@ public:
|
||||||
}
|
}
|
||||||
virtual QWeakPointer< TabBox::TabBoxClient > nextClientFocusChain(TabBox::TabBoxClient *client) const;
|
virtual QWeakPointer< TabBox::TabBoxClient > nextClientFocusChain(TabBox::TabBoxClient *client) const;
|
||||||
virtual QWeakPointer<TabBox::TabBoxClient> firstClientFocusChain() const;
|
virtual QWeakPointer<TabBox::TabBoxClient> firstClientFocusChain() const;
|
||||||
|
virtual bool isInFocusChain (TabBox::TabBoxClient* client) const;
|
||||||
virtual int nextDesktopFocusChain(int desktop) const {
|
virtual int nextDesktopFocusChain(int desktop) const {
|
||||||
Q_UNUSED(desktop)
|
Q_UNUSED(desktop)
|
||||||
return 1;
|
return 1;
|
||||||
|
|
|
@ -63,4 +63,24 @@ void TestTabBoxClientModel::testCreateClientListNoActiveClient()
|
||||||
QCOMPARE(clientModel->rowCount(), 2);
|
QCOMPARE(clientModel->rowCount(), 2);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void TestTabBoxClientModel::testCreateClientListActiveClientNotInFocusChain()
|
||||||
|
{
|
||||||
|
MockTabBoxHandler tabboxhandler;
|
||||||
|
tabboxhandler.setConfig(TabBox::TabBoxConfig());
|
||||||
|
TabBox::ClientModel *clientModel = new TabBox::ClientModel(&tabboxhandler);
|
||||||
|
// create two windows, rowCount() should go to two
|
||||||
|
QWeakPointer<TabBox::TabBoxClient> client = tabboxhandler.createMockWindow(QString("test"), 1);
|
||||||
|
client = tabboxhandler.createMockWindow(QString("test2"), 2);
|
||||||
|
clientModel->createClientList();
|
||||||
|
QCOMPARE(clientModel->rowCount(), 2);
|
||||||
|
|
||||||
|
// simulate that the active client is not in the focus chain
|
||||||
|
// for that we use the closeWindow of the MockTabBoxHandler which
|
||||||
|
// removes the Client from the Focus Chain but leaves the active window as it is
|
||||||
|
QSharedPointer<TabBox::TabBoxClient> clientOwner = client.toStrongRef();
|
||||||
|
tabboxhandler.closeWindow(client.data());
|
||||||
|
clientModel->createClientList();
|
||||||
|
QCOMPARE(clientModel->rowCount(), 1);
|
||||||
|
}
|
||||||
|
|
||||||
QTEST_MAIN(TestTabBoxClientModel)
|
QTEST_MAIN(TestTabBoxClientModel)
|
||||||
|
|
|
@ -40,6 +40,13 @@ private slots:
|
||||||
* See BUG: 305449
|
* See BUG: 305449
|
||||||
**/
|
**/
|
||||||
void testCreateClientListNoActiveClient();
|
void testCreateClientListNoActiveClient();
|
||||||
|
/**
|
||||||
|
* Tests the creation of the Client list for the case that
|
||||||
|
* the active Client is not in the Focus chain.
|
||||||
|
*
|
||||||
|
* See BUG: 306260
|
||||||
|
**/
|
||||||
|
void testCreateClientListActiveClientNotInFocusChain();
|
||||||
};
|
};
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
Loading…
Reference in a new issue