/***************************************************************** Copyright 2008 Christian Mollekopf Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions: The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software. THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. ******************************************************************/ #include "groupmanager.h" #include #include #include #include #include #include #include #include #include #include #include "abstractsortingstrategy.h" #include "startup.h" #include "task.h" #include "taskitem.h" #include "taskgroup.h" #include "taskmanager.h" #include "strategies/alphasortingstrategy.h" #include "strategies/desktopsortingstrategy.h" #include "strategies/programgroupingstrategy.h" #include "strategies/manualgroupingstrategy.h" #include "strategies/manualsortingstrategy.h" #include "launcheritem.h" #include "launcherconfig.h" namespace TaskManager { static const QByteArray pixmapFormat = QImageWriter::defaultImageFormat(); class GroupManagerPrivate { public: GroupManagerPrivate(GroupManager *manager) : q(manager), sortingStrategy(GroupManager::NoSorting), groupingStrategy(GroupManager::NoGrouping), lastGroupingStrategy(GroupManager::NoGrouping), abstractGroupingStrategy(0), abstractSortingStrategy(0), currentScreen(-1), groupIsFullLimit(0), showOnlyCurrentDesktop(true), showOnlyCurrentScreen(false), showOnlyMinimized(false), onlyGroupWhenFull(false), changingGroupingStrategy(false), readingLauncherConfig(false), separateLaunchers(true), forceGrouping(false), launchersLocked(false) { } /** reload all tasks from TaskManager */ void reloadTasks(); void actuallyReloadTasks(); /** * Keep track of changes in Taskmanager */ void currentDesktopChanged(int); void taskChanged(::TaskManager::Task *, ::TaskManager::TaskChanges); void checkScreenChange(); void taskDestroyed(QObject *item); void startupItemDestroyed(AbstractGroupableItem *); void checkIfFull(); void actuallyCheckIfFull(); bool addTask(::TaskManager::Task *); void removeTask(::TaskManager::Task *); void addStartup(::TaskManager::Startup *); void removeStartup(::TaskManager::Startup *); void sycocaChanged(const QStringList &types); void launcherVisibilityChange(); void checkLauncherVisibility(LauncherItem *launcher); void saveLauncher(LauncherItem *launcher); bool saveLauncher(LauncherItem *launcher, KConfigGroup &group); void unsaveLauncher(LauncherItem *launcher); void saveLauncherConfig(); void saveLauncherConfig(KConfigGroup &cg); QList readLauncherConfig(KConfigGroup &cg); int launcherIndex(const KUrl &url); KConfigGroup launcherConfig(const KConfigGroup &config = KConfigGroup()); TaskGroup *currentRootGroup(); GroupManager *q; QHash startupList; GroupManager::TaskSortingStrategy sortingStrategy; GroupManager::TaskGroupingStrategy groupingStrategy; GroupManager::TaskGroupingStrategy lastGroupingStrategy; AbstractGroupingStrategy *abstractGroupingStrategy; AbstractSortingStrategy *abstractSortingStrategy; int currentScreen; QTimer screenTimer; QTimer reloadTimer; QTimer checkIfFullTimer; QSet geometryTasks; int groupIsFullLimit; QHash rootGroups; //container for groups QList launchers; int currentDesktop; bool showOnlyCurrentDesktop : 1; bool showOnlyCurrentScreen : 1; bool showOnlyMinimized : 1; bool onlyGroupWhenFull : 1; bool changingGroupingStrategy : 1; bool readingLauncherConfig : 1; bool separateLaunchers : 1; bool forceGrouping : 1; bool launchersLocked : 1; }; GroupManager::GroupManager(QObject *parent) : QObject(parent), d(new GroupManagerPrivate(this)) { connect(TaskManager::self(), SIGNAL(taskAdded(::TaskManager::Task *)), this, SLOT(addTask(::TaskManager::Task *))); connect(TaskManager::self(), SIGNAL(taskRemoved(::TaskManager::Task *)), this, SLOT(removeTask(::TaskManager::Task *))); connect(TaskManager::self(), SIGNAL(startupAdded(::TaskManager::Startup *)), this, SLOT(addStartup(::TaskManager::Startup *))); connect(TaskManager::self(), SIGNAL(startupRemoved(::TaskManager::Startup *)), this, SLOT(removeStartup(::TaskManager::Startup *))); connect(KSycoca::self(), SIGNAL(databaseChanged(QStringList)), this, SLOT(sycocaChanged(const QStringList &))); d->currentDesktop = TaskManager::self()->currentDesktop(); d->rootGroups[d->currentDesktop] = new TaskGroup(this, "RootGroup"); d->reloadTimer.setSingleShot(true); d->reloadTimer.setInterval(0); connect(&d->reloadTimer, SIGNAL(timeout()), this, SLOT(actuallyReloadTasks())); d->screenTimer.setSingleShot(true); d->screenTimer.setInterval(100); connect(&d->screenTimer, SIGNAL(timeout()), this, SLOT(checkScreenChange())); d->checkIfFullTimer.setSingleShot(true); d->checkIfFullTimer.setInterval(0); connect(&d->checkIfFullTimer, SIGNAL(timeout()), this, SLOT(actuallyCheckIfFull())); } GroupManager::~GroupManager() { TaskManager::self()->setTrackGeometry(false); delete d->abstractSortingStrategy; delete d->abstractGroupingStrategy; delete d; } TaskGroup *GroupManagerPrivate::currentRootGroup() { return rootGroups[currentDesktop]; } void GroupManagerPrivate::reloadTasks() { reloadTimer.start(); } void GroupManagerPrivate::actuallyReloadTasks() { //kDebug() << "number of tasks available " << TaskManager::self()->tasks().size(); QHash taskList = TaskManager::self()->tasks(); QMutableHashIterator it(taskList); while (it.hasNext()) { it.next(); if (addTask(it.value())) { //kDebug() << "task added " << it.value()->visibleName(); it.remove(); } } // Remove what remains it.toFront(); while (it.hasNext()) { it.next(); removeTask(it.value()); } emit q->reload(); } void GroupManagerPrivate::addStartup(::TaskManager::Startup *task) { //kDebug(); if (!startupList.contains(task)) { TaskItem *item = new TaskItem(q, task); startupList.insert(task, item); currentRootGroup()->add(item); QObject::connect(item, SIGNAL(destroyed(AbstractGroupableItem*)), q, SLOT(startupItemDestroyed(AbstractGroupableItem*))); } } void GroupManagerPrivate::removeStartup(::TaskManager::Startup *task) { //kDebug(); if (!startupList.contains(task)) { kDebug() << "invalid startup task"; return; } TaskItem *item = startupList.take(task); if (item->parentGroup()) { item->parentGroup()->remove(item); } item->setTaskPointer(0); } bool GroupManagerPrivate::addTask(::TaskManager::Task *task) { if (!task) { return false; } //kDebug(); /* kDebug() << task->visibleName() << task->visibleNameWithState() << task->name() << task->className() << task->classClass(); */ bool skip = false; if (!task->showInTaskbar()) { //kDebug() << "Do not show in taskbar"; skip = true; } if (showOnlyCurrentScreen && !task->isOnScreen(currentScreen)) { //kDebug() << "Not on this screen and showOnlyCurrentScreen"; skip = true; } // Should the Task be displayed ? We always display if attention is demaded if (!task->demandsAttention()) { // As the Task doesn't demand attention // go through all filters whether the task should be displayed or not if (showOnlyCurrentDesktop && !task->isOnCurrentDesktop()) { /* kDebug() << "Not on this desktop and showOnlyCurrentDesktop" << KWindowSystem::currentDesktop() << task->desktop(); */ skip = true; } if (showOnlyMinimized && !task->isMinimized()) { //kDebug() << "Not minimized and only showing minimized"; skip = true; } NET::WindowType type = task->info().windowType(NET::NormalMask | NET::DialogMask | NET::UtilityMask); if (type == NET::Utility) { //kDebug() << "skipping utility window" << task->name(); skip = true; } //TODO: should we check for transiency? if so the following code can detect it. /* QHash ::iterator it = d->itemList.begin(); while (it != d->itemList.end()) { TaskItem *item = it.value(); if (item->task()->hasTransient(task->window())) { kDebug() << "TRANSIENT TRANSIENT TRANSIENT!"; return flase; } ++it; } */ } //Ok the Task should be displayed TaskItem *item = qobject_cast(currentRootGroup()->getMemberByWId(task->window())); if (!item || skip) { TaskItem *startupItem = 0; QHash::iterator it = startupList.begin(); QHash::iterator itEnd = startupList.end(); while (it != itEnd) { if (it.key()->matchesWindow(task->window())) { //kDebug() << "startup task found"; item = startupItem = it.value(); startupList.erase(it); QObject::disconnect(item, 0, q, 0); if (!skip) { item->setTaskPointer(task); } break; } ++it; } // if we are to skip because we don't display, we simply delete the startup related to it if (skip) { delete startupItem; return false; } if (!item) { item = new TaskItem(q, task); } QObject::connect(task, SIGNAL(destroyed(QObject*)), q, SLOT(taskDestroyed(QObject*))); foreach (LauncherItem * launcher, launchers) { if (launcher->associateItemIfMatches(item)) { // Task demands attention, so is to be shown, therefore hide the launcher... currentRootGroup()->remove(launcher); } } } //Find a fitting group for the task with GroupingStrategies if (abstractGroupingStrategy && (forceGrouping || !task->demandsAttention())) { //do not group attention tasks abstractGroupingStrategy->handleItem(item); } else { currentRootGroup()->add(item); if (abstractSortingStrategy) { abstractSortingStrategy->handleItem(item); abstractSortingStrategy->check(item); } } if (showOnlyCurrentScreen) { geometryTasks.insert(task); } return true; } void GroupManagerPrivate::removeTask(::TaskManager::Task *task) { if (!task) { return; } //kDebug() << "remove: " << task->visibleName(); geometryTasks.remove(task); AbstractGroupableItem *item = currentRootGroup()->getMemberByWId(task->window()); if (!item) { // this can happen if the window hasn't been caught previously, // of it it is an ignored type such as a NET::Utility type window //kDebug() << "invalid item"; return; } foreach (LauncherItem * launcher, launchers) { launcher->removeItemIfAssociated(item); } if (item->parentGroup()) { item->parentGroup()->remove(item); } //the item must exist as long as the Task does because of activate calls so don't delete the item here, it will delete itself. } void GroupManagerPrivate::taskDestroyed(QObject *item) { Task *task = static_cast(item); if (showOnlyCurrentScreen) { geometryTasks.remove(task); } } void GroupManagerPrivate::startupItemDestroyed(AbstractGroupableItem *item) { TaskItem *taskItem = static_cast(item); startupList.remove(startupList.key(taskItem)); geometryTasks.remove(taskItem->task()); } bool GroupManager::manualGroupingRequest(AbstractGroupableItem* item, TaskGroup* groupItem) { if (d->abstractGroupingStrategy) { return d->abstractGroupingStrategy->manualGroupingRequest(item, groupItem); } return false; } bool GroupManager::manualGroupingRequest(ItemList items) { if (d->abstractGroupingStrategy) { return d->abstractGroupingStrategy->manualGroupingRequest(items); } return false; } bool GroupManager::manualSortingRequest(AbstractGroupableItem* taskItem, int newIndex) { if (d->abstractSortingStrategy && (!launchersLocked() || separateLaunchers() || newIndex >= launcherCount())) { return d->abstractSortingStrategy->manualSortingRequest(taskItem, newIndex); } return false; } GroupPtr GroupManager::rootGroup() const { return d->currentRootGroup(); } void GroupManagerPrivate::currentDesktopChanged(int newDesktop) { //kDebug(); if (!showOnlyCurrentDesktop) { return; } if (currentDesktop == newDesktop) { return; } if (!rootGroups.contains(newDesktop)) { kDebug() << "created new desk group"; rootGroups[newDesktop] = new TaskGroup(q, "RootGroup"); if (abstractSortingStrategy) { abstractSortingStrategy->handleGroup(rootGroups[newDesktop]); } } if (onlyGroupWhenFull) { QObject::disconnect(currentRootGroup(), SIGNAL(itemAdded(AbstractGroupableItem*)), q, SLOT(checkIfFull())); QObject::disconnect(currentRootGroup(), SIGNAL(itemRemoved(AbstractGroupableItem*)), q, SLOT(checkIfFull())); } currentDesktop = newDesktop; foreach (LauncherItem * item, launchers) { if (item->shouldShow(q)) { rootGroups[currentDesktop]->add(item); } else { rootGroups[currentDesktop]->remove(item); } } if (onlyGroupWhenFull) { QObject::connect(currentRootGroup(), SIGNAL(itemAdded(AbstractGroupableItem*)), q, SLOT(checkIfFull())); QObject::connect(currentRootGroup(), SIGNAL(itemRemoved(AbstractGroupableItem*)), q, SLOT(checkIfFull())); } actuallyReloadTasks(); } void GroupManagerPrivate::taskChanged(::TaskManager::Task *task, ::TaskManager::TaskChanges changes) { //kDebug(); if (!task) { return; } bool takeAction = false; bool show = true; if (showOnlyCurrentDesktop && changes & ::TaskManager::DesktopChanged) { takeAction = true; show = task->isOnCurrentDesktop(); //kDebug() << task->visibleName() << "on" << TaskManager::self()->currentDesktop(); } if (showOnlyMinimized && changes & ::TaskManager::StateChanged) { //TODO: wouldn't it be nice to get notification of JUST minimization? takeAction = true; show = task->isMinimized(); } if (showOnlyCurrentScreen && changes & ::TaskManager::GeometryChanged) { geometryTasks.insert(task); if (!screenTimer.isActive()) { screenTimer.start(); } } if (changes & ::TaskManager::AttentionChanged) { // we show tasks anyway if they demand attention // so whenever our state changes ... try to re-adjust it takeAction = true; show = true; } // Some apps, eg. LibreOffice, change classClass/className after start-up... if (changes & ::TaskManager::ClassChanged) { // Instead of just moving item (as what happend in the #if below), just remove and re-add the task. // This way the grouping happens properly. AbstractGroupableItem *item = currentRootGroup()->getMemberByWId(task->window()); if (item && TaskItemType == item->itemType()) { static_cast(item)->resetLauncherCheck(); } removeTask(task); addTask(task); } if (!takeAction) { return; } show = show && (!showOnlyCurrentScreen || task->isOnScreen(currentScreen)); if (show) { //kDebug() << "add(task);"; addTask(task); } else { //kDebug() << "remove(task);"; removeTask(task); } } void GroupManager::setScreen(int screen) { //kDebug() << "new Screen: " << screen; d->currentScreen = screen; d->reloadTasks(); } int GroupManager::screen() const { return d->currentScreen; } void GroupManagerPrivate::checkScreenChange() { //kDebug(); if (showOnlyCurrentScreen) { foreach (Task *task, geometryTasks) { if (task->isOnScreen(currentScreen)) { addTask(task); } else { removeTask(task); } } } geometryTasks.clear(); } void GroupManager::reconnect() { //kDebug(); disconnect(TaskManager::self(), SIGNAL(desktopChanged(int)), this, SLOT(currentDesktopChanged(int))); disconnect(TaskManager::self(), SIGNAL(windowChanged(::TaskManager::Task *, ::TaskManager::TaskChanges)), this, SLOT(taskChanged(::TaskManager::Task *, ::TaskManager::TaskChanges))); if (d->showOnlyCurrentDesktop || d->showOnlyMinimized || d->showOnlyCurrentScreen) { // listen to the relevant task manager signals if (d->showOnlyCurrentDesktop) { connect(TaskManager::self(), SIGNAL(desktopChanged(int)), this, SLOT(currentDesktopChanged(int))); } connect(TaskManager::self(), SIGNAL(windowChanged(::TaskManager::Task *, ::TaskManager::TaskChanges)), this, SLOT(taskChanged(::TaskManager::Task *, ::TaskManager::TaskChanges))); } TaskManager::self()->setTrackGeometry(d->showOnlyCurrentScreen); if (!d->showOnlyCurrentScreen) { d->geometryTasks.clear(); } d->reloadTasks(); } KConfigGroup GroupManager::config() const { return KConfigGroup(); } bool GroupManager::addLauncher(const KUrl &url, const QIcon &icon, const QString &name, const QString &genericName, const QString &wmClass, int insertPos) { if (url.isEmpty() || launchersLocked()) { return false; } int index = launcherIndex(url); LauncherItem *launcher = -1 != index ? d->launchers.at(index) : 0L; // Do not insert launchers twice if (!launcher) { launcher = new LauncherItem(d->currentRootGroup(), url); if (!launcher->isValid()) { delete launcher; return false; } if (!icon.isNull()) { launcher->setIcon(icon); } if (!name.isEmpty()) { launcher->setName(name); } if (!genericName.isEmpty()) { launcher->setGenericName(genericName); } if (!wmClass.isEmpty()) { launcher->setWmClass(wmClass); } QStack groups; groups.push(d->currentRootGroup()); while (!groups.isEmpty()) { TaskGroup *group = groups.pop(); foreach (AbstractGroupableItem * item, group->members()) { if (item->itemType() == GroupItemType) { groups.push(static_cast(item)); } else { launcher->associateItemIfMatches(item); } } } if (insertPos >= 0 && insertPos < d->launchers.count()) { d->launchers.insert(insertPos, launcher); } else { d->launchers.append(launcher); } d->saveLauncherConfig(); d->saveLauncher(launcher); connect(launcher, SIGNAL(associationChanged()), this, SLOT(launcherVisibilityChange())); d->checkLauncherVisibility(launcher); if (!d->separateLaunchers && d->abstractSortingStrategy && ManualSorting == d->abstractSortingStrategy->type()) { // Ensure item is placed where launcher would be... foreach (AbstractGroupableItem * item, d->rootGroups[d->currentDesktop]->members()) { if (LauncherItemType != item->itemType() && item->launcherUrl() == url) { manualSortingRequest(item, launcherIndex(url)); break; } } if (!d->readingLauncherConfig) { emit launchersChanged(); } } } else if (d->readingLauncherConfig && !KDesktopFile::isDesktopFile(url.toLocalFile())) { // We are reading in config, and have already added this launcher (via the launcher list config). HWoever, this // is for an mebdded non-desktop launcher - so we set the details here... if (!icon.isNull()) { launcher->setIcon(icon); } if (!name.isEmpty()) { launcher->setName(name); } if (!genericName.isEmpty()) { launcher->setGenericName(genericName); } if (!wmClass.isEmpty()) { launcher->setWmClass(wmClass); } } return launcher; } void GroupManager::removeLauncher(const KUrl &url) { if (launchersLocked()) { return; } int index = launcherIndex(url); if (index == -1 || index >= d->launchers.count()) { return; } LauncherItem *launcher = d->launchers.at(index); if (!launcher) { return; } d->launchers.removeAt(index); d->saveLauncherConfig(); foreach (TaskGroup * rootGroup, d->rootGroups) { rootGroup->remove(launcher); } d->unsaveLauncher(launcher); launcher->deleteLater(); if (!d->separateLaunchers && d->abstractSortingStrategy && ManualSorting == d->abstractSortingStrategy->type()) { // Ensure item is placed at end of launchers... foreach (AbstractGroupableItem * item, d->rootGroups[d->currentDesktop]->members()) { if (LauncherItemType != item->itemType() && item->launcherUrl() == url) { manualSortingRequest(item, d->launchers.count()); break; } } if (!d->readingLauncherConfig) { emit launchersChanged(); } } } void GroupManagerPrivate::sycocaChanged(const QStringList &types) { if (types.contains("apps") || types.contains("xdgdata-apps")) { KUrl::List removals; foreach (LauncherItem *launcher, launchers) { if (!QFile::exists(launcher->launcherUrl().toLocalFile())) { removals << launcher->launcherUrl(); } } foreach (const KUrl & url, removals) { q->removeLauncher(url); } } } void GroupManagerPrivate::launcherVisibilityChange() { checkLauncherVisibility(qobject_cast(q->sender())); } void GroupManagerPrivate::checkLauncherVisibility(LauncherItem *launcher) { if (!launcher) { return; } if (launcher->shouldShow(q)) { rootGroups[currentDesktop]->add(launcher); } else { rootGroups[currentDesktop]->remove(launcher); } } bool GroupManager::launcherExists(const KUrl &url) const { return d->launcherIndex(url) != -1; } void GroupManager::readLauncherConfig(const KConfigGroup &cg) { KConfigGroup conf = d->launcherConfig(cg); if (!conf.isValid()) { return; } QList launchers = d->readLauncherConfig(conf); // prevents re-writing the results out d->readingLauncherConfig = true; QSet urls; foreach (KUrl l, launchers) { QString wmClass(l.queryItemValue("wmClass")); l.setQuery(QString()); if (addLauncher(l, QIcon(), QString(), QString(), wmClass)) { urls << l; } } foreach (const QString &key, conf.keyList()) { if ("Items" == key) { continue; } QStringList item = conf.readEntry(key, QStringList()); if (item.length() >= 4) { KUrl url(item.at(0)); KIcon icon; if (!item.at(1).isEmpty()) { icon = KIcon(item.at(1)); } else if (item.length() >= 5) { QPixmap pixmap; QByteArray bytes = QByteArray::fromBase64(item.at(4).toAscii()); pixmap.loadFromData(bytes, pixmapFormat); icon.addPixmap(pixmap); } QString name(item.at(2)); QString genericName(item.at(3)); if (addLauncher(url, icon, name, genericName, genericName)) { urls << url; } } } // a bit paranoiac, perhaps, but we check the removals first and then // remove the launchers after that scan because Qt's iterators operate // on a copy of the list and/or don't like multiple iterators messing // with the same list. we might get away with just calling removeLauncher // immediately without the removals KUrl::List, but this is known safe // and not a performance bottleneck KUrl::List removals; foreach (LauncherItem *launcher, d->launchers) { if (!urls.contains(launcher->launcherUrl())) { removals << launcher->launcherUrl(); } } foreach (const KUrl & url, removals) { removeLauncher(url); } d->readingLauncherConfig = false; } void GroupManager::exportLauncherConfig(const KConfigGroup &cg) { KConfigGroup conf = d->launcherConfig(cg); if (!conf.isValid()) { return; } foreach (LauncherItem * launcher, d->launchers) { d->saveLauncher(launcher, conf); } } int GroupManager::launcherIndex(const KUrl &url) const { return d->launcherIndex(url); } int GroupManager::launcherCount() const { return d->launchers.count(); } bool GroupManager::launchersLocked() const { return d->launchersLocked; } void GroupManager::setLaunchersLocked(bool l) { d->launchersLocked = l; } KUrl GroupManager::launcherForWmClass(const QString &wmClass) const { foreach (LauncherItem * l, d->launchers) { if (l->wmClass() == wmClass) { return l->launcherUrl(); } } return KUrl(); } QString GroupManager::launcherWmClass(const KUrl &url) const { int index = launcherIndex(url); LauncherItem *l = -1 != index ? d->launchers.at(index) : 0L; return l ? l->wmClass() : QString(); } bool GroupManager::isItemAssociatedWithLauncher(AbstractGroupableItem *item) const { if (item) { switch (item->itemType()) { case LauncherItemType: return true; case GroupItemType: { foreach (AbstractGroupableItem * i, static_cast(item)->members()) { if (isItemAssociatedWithLauncher(i)) { return true; } } break; } case TaskItemType: { foreach (LauncherItem * launcher, d->launchers) { if (launcher->isAssociated(item)) { return true; } } } } } return false; } void GroupManager::moveLauncher(const KUrl &url, int newIndex) { if (!url.isValid()) { return; } int oldIndex = launcherIndex(url); if (oldIndex >= 0 && newIndex != oldIndex) { d->launchers.insert(newIndex, d->launchers.takeAt(oldIndex)); d->saveLauncherConfig(); } } bool GroupManager::separateLaunchers() const { return d->separateLaunchers; } void GroupManager::setSeparateLaunchers(bool s) { d->separateLaunchers = s; } bool GroupManager::forceGrouping() const { return d->forceGrouping; } void GroupManager::setForceGrouping(bool s) { d->forceGrouping = s; } void GroupManager::createConfigurationInterface(KConfigDialog *parent) { new LauncherConfig(parent); } KConfigGroup GroupManagerPrivate::launcherConfig(const KConfigGroup &config) { KConfigGroup cg = config.isValid() ? config : q->config(); if (!cg.isValid()) { return cg; } return KConfigGroup(&cg, "Launchers"); } void GroupManagerPrivate::saveLauncher(LauncherItem *launcher) { if (readingLauncherConfig) { return; } KConfigGroup cg = launcherConfig(); if (!cg.isValid()) { return; } if (saveLauncher(launcher, cg)) { emit q->configChanged(); } } bool GroupManagerPrivate::saveLauncher(LauncherItem *launcher, KConfigGroup &cg) { // Dont save .desktop file launchers, as these are already stored in the launcher list... if (launcher->launcherUrl().isValid() && KDesktopFile::isDesktopFile(launcher->launcherUrl().toLocalFile())) { return false; } QVariantList launcherProperties; launcherProperties.append(launcher->launcherUrl().url()); launcherProperties.append(launcher->icon().name()); launcherProperties.append(launcher->name()); launcherProperties.append(launcher->genericName()); if (launcher->icon().name().isEmpty()) { QPixmap pixmap = launcher->icon().pixmap(QSize(64, 64)); QByteArray bytes; QBuffer buffer(&bytes); buffer.open(QIODevice::WriteOnly); pixmap.save(&buffer, pixmapFormat); launcherProperties.append(bytes.toBase64()); } cg.writeEntry(launcher->name(), launcherProperties); return true; } void GroupManagerPrivate::unsaveLauncher(LauncherItem *launcher) { KConfigGroup cg = launcherConfig(); if (!cg.isValid()) { return; } QString launcherKey = launcher->name(); if (cg.hasKey(launcherKey)) { cg.deleteEntry(launcherKey); emit q->configChanged(); } } void GroupManagerPrivate::saveLauncherConfig() { if (readingLauncherConfig) { return; } KConfigGroup cg = launcherConfig(); if (!cg.isValid()) { return; } saveLauncherConfig(cg); } void GroupManagerPrivate::saveLauncherConfig(KConfigGroup &cg) { QStringList details; foreach (LauncherItem * l, launchers) { KUrl u(l->launcherUrl()); if (!u.isEmpty()) { if (!l->wmClass().isEmpty()) { u.addQueryItem("wmClass", l->wmClass()); } details.append(u.url()); } } cg.writeEntry("Items", details); emit q->configChanged(); } QList GroupManagerPrivate::readLauncherConfig(KConfigGroup &cg) { QStringList items = cg.readEntry("Items", QStringList()); QList details; foreach (QString item, items) { if (!item.isEmpty()) { details.append(KUrl(item)); } } return details; } int GroupManagerPrivate::launcherIndex(const KUrl &url) { // we check first for exact matches ... int index = 0; foreach (const LauncherItem * item, launchers) { if (item->launcherUrl() == url) { return index; } ++index; } return -1; } bool GroupManager::onlyGroupWhenFull() const { return d->onlyGroupWhenFull; } void GroupManager::setOnlyGroupWhenFull(bool onlyGroupWhenFull) { //kDebug() << onlyGroupWhenFull; if (d->onlyGroupWhenFull == onlyGroupWhenFull) { return; } d->onlyGroupWhenFull = onlyGroupWhenFull; disconnect(d->currentRootGroup(), SIGNAL(itemAdded(AbstractGroupableItem*)), this, SLOT(checkIfFull())); disconnect(d->currentRootGroup(), SIGNAL(itemRemoved(AbstractGroupableItem*)), this, SLOT(checkIfFull())); if (onlyGroupWhenFull) { connect(d->currentRootGroup(), SIGNAL(itemAdded(AbstractGroupableItem*)), this, SLOT(checkIfFull())); connect(d->currentRootGroup(), SIGNAL(itemRemoved(AbstractGroupableItem*)), this, SLOT(checkIfFull())); d->checkIfFull(); } else { setGroupingStrategy(d->groupingStrategy); } } void GroupManager::setFullLimit(int limit) { //kDebug() << limit; d->groupIsFullLimit = limit; if (d->onlyGroupWhenFull) { d->checkIfFull(); } } void GroupManagerPrivate::checkIfFull() { // Start a timer so that if we have been triggered by a layouting // we give time for it to finish up instead of starting a new one // right away. checkIfFullTimer.start(); } void GroupManagerPrivate::actuallyCheckIfFull() { //kDebug(); if (!onlyGroupWhenFull || groupingStrategy != GroupManager::ProgramGrouping || changingGroupingStrategy) { return; } if (currentRootGroup()->totalSize() >= groupIsFullLimit) { if (!abstractGroupingStrategy) { geometryTasks.clear(); q->setGroupingStrategy(GroupManager::ProgramGrouping); } } else if (abstractGroupingStrategy) { geometryTasks.clear(); q->setGroupingStrategy(GroupManager::NoGrouping); //let the visualization think we still use the programGrouping groupingStrategy = GroupManager::ProgramGrouping; } } bool GroupManager::showOnlyCurrentScreen() const { return d->showOnlyCurrentScreen; } void GroupManager::setShowOnlyCurrentScreen(bool showOnlyCurrentScreen) { d->showOnlyCurrentScreen = showOnlyCurrentScreen; reconnect(); } bool GroupManager::showOnlyCurrentDesktop() const { return d->showOnlyCurrentDesktop; } void GroupManager::setShowOnlyCurrentDesktop(bool showOnlyCurrentDesktop) { d->showOnlyCurrentDesktop = showOnlyCurrentDesktop; reconnect(); } bool GroupManager::showOnlyMinimized() const { return d->showOnlyMinimized; } void GroupManager::setShowOnlyMinimized(bool showOnlyMinimized) { d->showOnlyMinimized = showOnlyMinimized; reconnect(); } GroupManager::TaskSortingStrategy GroupManager::sortingStrategy() const { return d->sortingStrategy; } AbstractSortingStrategy* GroupManager::taskSorter() const { return d->abstractSortingStrategy; } void GroupManager::setSortingStrategy(TaskSortingStrategy sortOrder) { //kDebug() << sortOrder; if (d->abstractSortingStrategy) { if (d->abstractSortingStrategy->type() == sortOrder) { return; } d->abstractSortingStrategy->deleteLater(); d->abstractSortingStrategy = 0; } switch (sortOrder) { case ManualSorting: d->abstractSortingStrategy = new ManualSortingStrategy(this); break; case AlphaSorting: d->abstractSortingStrategy = new AlphaSortingStrategy(this); break; case DesktopSorting: d->abstractSortingStrategy = new DesktopSortingStrategy(this); break; case NoSorting: //manual and no grouping result both in non automatic grouping break; default: kDebug() << "Invalid Strategy"; } if (d->abstractSortingStrategy) { foreach (TaskGroup * group, d->rootGroups) { d->abstractSortingStrategy->handleGroup(group); } } d->sortingStrategy = sortOrder; reconnect(); } GroupManager::TaskGroupingStrategy GroupManager::groupingStrategy() const { return d->groupingStrategy; } AbstractGroupingStrategy* GroupManager::taskGrouper() const { return d->abstractGroupingStrategy; } void GroupManager::setGroupingStrategy(TaskGroupingStrategy strategy) { if (d->changingGroupingStrategy || (d->abstractGroupingStrategy && d->abstractGroupingStrategy->type() == strategy)) { return; } d->changingGroupingStrategy = true; //kDebug() << strategy << kBacktrace(); if (d->onlyGroupWhenFull) { disconnect(d->currentRootGroup(), SIGNAL(itemAdded(AbstractGroupableItem*)), this, SLOT(checkIfFull())); disconnect(d->currentRootGroup(), SIGNAL(itemRemoved(AbstractGroupableItem*)), this, SLOT(checkIfFull())); } if (d->abstractGroupingStrategy) { disconnect(d->abstractGroupingStrategy, 0, this, 0); d->abstractGroupingStrategy->destroy(); d->abstractGroupingStrategy = 0; } switch (strategy) { case ManualGrouping: d->abstractGroupingStrategy = new ManualGroupingStrategy(this); break; case ProgramGrouping: d->abstractGroupingStrategy = new ProgramGroupingStrategy(this); break; case NoGrouping: break; default: kDebug() << "Strategy not implemented"; } d->groupingStrategy = strategy; d->actuallyReloadTasks(); if (d->onlyGroupWhenFull) { connect(d->currentRootGroup(), SIGNAL(itemAdded(AbstractGroupableItem*)), this, SLOT(checkIfFull())); connect(d->currentRootGroup(), SIGNAL(itemRemoved(AbstractGroupableItem*)), this, SLOT(checkIfFull())); } d->changingGroupingStrategy = false; } } // TaskManager namespace #include "moc_groupmanager.cpp"