kwin/tabbox/clientmodel.cpp

276 lines
7.8 KiB
C++
Raw Normal View History

/********************************************************************
KWin - the KDE window manager
This file is part of the KDE project.
Copyright (C) 2009 Martin Gräßlin <kde@martin-graesslin.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/>.
*********************************************************************/
// own
#include "clientmodel.h"
// tabbox
#include "tabboxconfig.h"
#include "tabboxhandler.h"
// Qt
#include <QTextStream>
// KDE
#include <KLocale>
// other
#include <math.h>
namespace KWin
{
namespace TabBox
{
2011-01-30 14:34:42 +00:00
ClientModel::ClientModel(QObject* parent)
: QAbstractItemModel(parent)
{
QHash<int, QByteArray> roles;
roles[CaptionRole] = "caption";
roles[DesktopNameRole] = "desktopName";
roles[MinimizedRole] = "minimized";
roles[WIdRole] = "windowId";
roles[CloseableRole] = "closeable";
setRoleNames(roles);
2011-01-30 14:34:42 +00:00
}
ClientModel::~ClientModel()
2011-01-30 14:34:42 +00:00
{
}
2011-01-30 14:34:42 +00:00
QVariant ClientModel::data(const QModelIndex& index, int role) const
{
if (!index.isValid())
return QVariant();
2011-01-30 14:34:42 +00:00
if (m_clientList.isEmpty()) {
return QVariant();
2011-01-30 14:34:42 +00:00
}
2011-01-30 14:34:42 +00:00
int clientIndex = index.row() * columnCount() + index.column();
if (clientIndex >= m_clientList.count())
return QVariant();
QSharedPointer<TabBoxClient> client = m_clientList[ clientIndex ].toStrongRef();
if (!client) {
return QVariant();
}
2011-01-30 14:34:42 +00:00
switch(role) {
case Qt::DisplayRole:
case CaptionRole:
return client->caption();
2011-01-30 14:34:42 +00:00
case ClientRole:
return qVariantFromValue((void*)client.data());
2011-01-30 14:34:42 +00:00
case DesktopNameRole: {
return tabBox->desktopName(client.data());
}
2011-01-30 14:34:42 +00:00
case WIdRole:
return qulonglong(client->window());
2011-01-30 14:34:42 +00:00
case MinimizedRole:
return client->isMinimized();
case CloseableRole:
//clients that claim to be first are not closeable
return client->isCloseable() && !client->isFirstInTabBox();
2011-01-30 14:34:42 +00:00
default:
return QVariant();
}
}
QString ClientModel::longestCaption() const
{
QString caption;
foreach (QWeakPointer<TabBoxClient> clientPointer, m_clientList) {
QSharedPointer<TabBoxClient> client = clientPointer.toStrongRef();
if (!client) {
continue;
}
if (client->caption().size() > caption.size()) {
caption = client->caption();
}
}
return caption;
}
2011-01-30 14:34:42 +00:00
int ClientModel::columnCount(const QModelIndex& parent) const
{
Q_UNUSED(parent)
int count = 1;
2011-01-30 14:34:42 +00:00
switch(tabBox->config().layout()) {
case TabBoxConfig::HorizontalLayout:
count = m_clientList.count();
break;
case TabBoxConfig::VerticalLayout:
count = 1;
break;
case TabBoxConfig::HorizontalVerticalLayout:
count = qRound(sqrt(float(m_clientList.count())));
if (count * count < m_clientList.count())
count++;
break;
}
2011-01-30 14:34:42 +00:00
return qMax(count, 1);
}
2011-01-30 14:34:42 +00:00
int ClientModel::rowCount(const QModelIndex& parent) const
{
Q_UNUSED(parent)
int count = 1;
2011-01-30 14:34:42 +00:00
switch(tabBox->config().layout()) {
case TabBoxConfig::HorizontalLayout:
count = 1;
break;
case TabBoxConfig::VerticalLayout:
count = m_clientList.count();
break;
case TabBoxConfig::HorizontalVerticalLayout:
count = qRound(sqrt(float(m_clientList.count())));
break;
}
return count;
2011-01-30 14:34:42 +00:00
}
2011-01-30 14:34:42 +00:00
QModelIndex ClientModel::parent(const QModelIndex& child) const
{
Q_UNUSED(child)
return QModelIndex();
2011-01-30 14:34:42 +00:00
}
2011-01-30 14:34:42 +00:00
QModelIndex ClientModel::index(int row, int column, const QModelIndex& parent) const
{
Q_UNUSED(parent)
int index = row * columnCount() + column;
2011-01-30 14:34:42 +00:00
if (index >= m_clientList.count() && !m_clientList.isEmpty())
return QModelIndex();
2011-01-30 14:34:42 +00:00
return createIndex(row, column);
}
QModelIndex ClientModel::index(QWeakPointer<TabBoxClient> client) const
2011-01-30 14:34:42 +00:00
{
if (!m_clientList.contains(client))
return QModelIndex();
2011-01-30 14:34:42 +00:00
int index = m_clientList.indexOf(client);
int row = index / columnCount();
int column = index % columnCount();
2011-01-30 14:34:42 +00:00
return createIndex(row, column);
}
2011-01-30 14:34:42 +00:00
void ClientModel::createClientList(bool partialReset)
{
createClientList(tabBox->currentDesktop(), partialReset);
}
2011-01-30 14:34:42 +00:00
void ClientModel::createClientList(int desktop, bool partialReset)
{
TabBoxClient* start = tabBox->activeClient().toStrongRef().data();
// TODO: new clients are not added at correct position
if (partialReset && !m_clientList.isEmpty()) {
QSharedPointer<TabBoxClient> firstClient = m_clientList.first().toStrongRef();
if (firstClient) {
start = firstClient.data();
}
}
m_clientList.clear();
QList< QWeakPointer< TabBoxClient > > stickyClients;
2011-01-30 14:34:42 +00:00
switch(tabBox->config().clientSwitchingMode()) {
case TabBoxConfig::FocusChainSwitching: {
TabBoxClient* c = start;
if (!tabBox->isInFocusChain(c)) {
QSharedPointer<TabBoxClient> firstClient = tabBox->firstClientFocusChain().toStrongRef();
if (firstClient) {
c = firstClient.data();
}
}
2011-01-30 14:34:42 +00:00
TabBoxClient* stop = c;
do {
QWeakPointer<TabBoxClient> add = tabBox->clientToAddToList(c, desktop);
if (!add.isNull()) {
m_clientList += add;
if (add.data()->isFirstInTabBox()) {
stickyClients << add;
}
}
c = tabBox->nextClientFocusChain(c).data();
} while (c && c != stop);
2011-01-30 14:34:42 +00:00
break;
}
case TabBoxConfig::StackingOrderSwitching: {
// TODO: needs improvement
TabBoxClientList stacking = tabBox->stackingOrder();
TabBoxClient* c = stacking.first().data();
2011-01-30 14:34:42 +00:00
TabBoxClient* stop = c;
int index = 0;
while (c) {
QWeakPointer<TabBoxClient> add = tabBox->clientToAddToList(c, desktop);
if (!add.isNull()) {
if (start == add.data()) {
2011-01-30 14:34:42 +00:00
m_clientList.removeAll(add);
m_clientList.prepend(add);
} else
m_clientList += add;
if (add.data()->isFirstInTabBox()) {
stickyClients << add;
}
2011-01-30 14:34:42 +00:00
}
if (index >= stacking.size() - 1) {
c = NULL;
} else {
c = stacking[++index].data();
}
2011-01-30 14:34:42 +00:00
if (c == stop)
break;
}
2011-01-30 14:34:42 +00:00
break;
}
}
foreach (QWeakPointer< TabBoxClient > c, stickyClients) {
m_clientList.removeAll(c);
m_clientList.prepend(c);
}
if (tabBox->config().showDesktopMode() == TabBoxConfig::ShowDesktopClient || m_clientList.isEmpty()) {
QWeakPointer<TabBoxClient> desktopClient = tabBox->desktopClient();
if (!desktopClient.isNull())
2011-01-30 14:34:42 +00:00
m_clientList.append(desktopClient);
}
2011-01-30 14:34:42 +00:00
reset();
}
void ClientModel::close(int i)
{
QModelIndex ind = index(i, 0);
if (!ind.isValid()) {
return;
}
QSharedPointer<TabBoxClient> client = m_clientList.at(i).toStrongRef();
if (client) {
client->close();
}
}
void ClientModel::activate(int i)
{
QModelIndex ind = index(i, 0);
if (!ind.isValid()) {
return;
}
tabBox->setCurrentIndex(ind);
tabBox->activateAndClose();
}
} // namespace Tabbox
} // namespace KWin