/***************************************************************** Copyright (c) 2001 Matthias Elter <elter@kde.org> Copyright (c) 2004 Sebastian Wolff Copyright (c) 2005 Aaron Seigo <aseigo@kde.org> 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. ***************************************************************** Additional changes: - 2013/10/22 Michele Calgaro * added support for display mode (Icons and Text, Text only, Icons only) and removed "Show application icons" */ #include <math.h> #include <tqapplication.h> #include <tqbitmap.h> #include <tqdesktopwidget.h> #include <tqlayout.h> #include <tqpainter.h> #include <tqstringlist.h> #include <dcopclient.h> #include <tdeapplication.h> #include <kdebug.h> #include <tdeglobal.h> #include <kglobalaccel.h> #include <kimageeffect.h> #include <tdelocale.h> #include <kstandarddirs.h> #include "kickerSettings.h" #include "taskbarsettings.h" #include "taskcontainer.h" #include "taskmanager.h" #include "taskbar.h" #include "taskbar.moc" #define READ_MERGED_TASBKAR_SETTING(x) ((m_settingsObject->useGlobalSettings())?m_globalSettingsObject->x():m_settingsObject->x()) TaskBar::TaskBar( TaskBarSettings* settingsObject, TaskBarSettings* globalSettingsObject, TQWidget *parent, const char *name ) : Panner( parent, name ), m_showAllWindows(false), m_cycleWheel(false), m_currentScreen(-1), m_showOnlyCurrentScreen(false), m_sortByDesktop(false), m_displayIconsNText(settingsObject->DisplayIconsAndText), m_showOnlyIconified(false), m_showTaskStates(0), m_textShadowEngine(0), m_ignoreUpdates(false), m_relayoutTimer(0, "TaskBar::m_relayoutTimer") { arrowType = LeftArrow; blocklayout = true; m_settingsObject = settingsObject; if (m_settingsObject) { m_settingsObject->readConfig(); } m_globalSettingsObject = globalSettingsObject; if (m_globalSettingsObject) { m_globalSettingsObject->readConfig(); } // init setSizePolicy( TQSizePolicy( TQSizePolicy::Expanding, TQSizePolicy::Expanding ) ); m_sortByAppPrev = READ_MERGED_TASBKAR_SETTING(sortByApp); // setup animation frames for (int i = 1; i < 11; i++) { frames.append(new TQPixmap(locate("data", "kicker/pics/disk" + TQString::number(i) + ".png"))); } // configure configure(); connect(&m_relayoutTimer, TQT_SIGNAL(timeout()), this, TQT_SLOT(reLayout())); connect(this, TQT_SIGNAL(contentsMoving(int, int)), TQT_SLOT(setBackground())); // connect manager connect(TaskManager::the(), TQT_SIGNAL(taskAdded(Task::Ptr)), this, TQT_SLOT(add(Task::Ptr))); connect(TaskManager::the(), TQT_SIGNAL(taskRemoved(Task::Ptr)), this, TQT_SLOT(remove(Task::Ptr))); connect(TaskManager::the(), TQT_SIGNAL(startupAdded(Startup::Ptr)), this, TQT_SLOT(add(Startup::Ptr))); connect(TaskManager::the(), TQT_SIGNAL(startupRemoved(Startup::Ptr)), this, TQT_SLOT(remove(Startup::Ptr))); connect(TaskManager::the(), TQT_SIGNAL(desktopChanged(int)), this, TQT_SLOT(desktopChanged(int))); connect(TaskManager::the(), TQT_SIGNAL(windowChanged(Task::Ptr)), this, TQT_SLOT(windowChanged(Task::Ptr))); isGrouping = shouldGroup(); // register existant tasks Task::Dict tasks = TaskManager::the()->tasks(); Task::Dict::iterator taskEnd = tasks.end(); for (Task::Dict::iterator it = tasks.begin(); it != taskEnd; ++it) { add(it.data()); } // register existant startups Startup::List startups = TaskManager::the()->startups(); Startup::List::iterator startupEnd = startups.end(); for (Startup::List::iterator sIt = startups.begin(); sIt != startupEnd; ++sIt) { add((*sIt)); } blocklayout = false; connect(kapp, TQT_SIGNAL(settingsChanged(int)), TQT_SLOT(slotSettingsChanged(int))); keys = new TDEGlobalAccel( TQT_TQOBJECT(this) ); #include "taskbarbindings.cpp" keys->readSettings(); keys->updateConnections(); reLayout(); } TaskBar::~TaskBar() { for (TaskContainer::Iterator it = m_hiddenContainers.begin(); it != m_hiddenContainers.end(); ++it) { (*it)->deleteLater(); } for (TaskContainer::List::const_iterator it = containers.constBegin(); it != containers.constEnd(); ++it) { (*it)->deleteLater(); } for (PixmapList::const_iterator it = frames.constBegin(); it != frames.constEnd(); ++it) { delete *it; } delete m_textShadowEngine; } KTextShadowEngine *TaskBar::textShadowEngine() { if (!m_textShadowEngine) m_textShadowEngine = new KTextShadowEngine(); return m_textShadowEngine; } TQSize TaskBar::sizeHint() const { // get our minimum height based on the minimum button height or the // height of the font in use, which is largest TQFontMetrics fm(TDEGlobalSettings::taskbarFont()); int minButtonHeight = fm.height() > READ_MERGED_TASBKAR_SETTING(minimumButtonHeight) ? fm.height() : READ_MERGED_TASBKAR_SETTING(minimumButtonHeight); return TQSize(BUTTON_MIN_WIDTH, minButtonHeight); } TQSize TaskBar::sizeHint( KPanelExtension::Position p, TQSize maxSize) const { // get our minimum height based on the minimum button height or the // height of the font in use, which is largest TQFontMetrics fm(TDEGlobalSettings::taskbarFont()); int minButtonHeight = fm.height() > READ_MERGED_TASBKAR_SETTING(minimumButtonHeight) ? fm.height() : READ_MERGED_TASBKAR_SETTING(minimumButtonHeight); if ( p == KPanelExtension::Left || p == KPanelExtension::Right ) { // Vertical layout // Minimum space allows for one icon, the window list button and the up/down scrollers int minHeight = minButtonHeight*3; if (minHeight > maxSize.height()) return maxSize; return TQSize(maxSize.width(), minHeight); } else { // Horizontal layout // Minimum space allows for one column of icons, the window list button and the left/right scrollers int min_width=BUTTON_MIN_WIDTH*3; if (min_width > maxSize.width()) return maxSize; return TQSize(min_width, maxSize.height()); } } bool TaskBar::showIcons() const { return (m_displayIconsNText==m_settingsObject->DisplayIconsAndText || m_displayIconsNText==m_settingsObject->DisplayIconsOnly); } bool TaskBar::showText() const { return (m_displayIconsNText==m_settingsObject->DisplayIconsAndText || m_displayIconsNText==m_settingsObject->DisplayTextOnly); } void TaskBar::configure() { bool wasShowWindows = m_showAllWindows; bool wasSortByDesktop = m_sortByDesktop; bool wasCycleWheel = m_cycleWheel; bool wasDisplayIconsNText = m_displayIconsNText; bool wasShowOnlyIconified = m_showOnlyIconified; int wasShowTaskStates = m_showTaskStates; m_showAllWindows = READ_MERGED_TASBKAR_SETTING(showAllWindows); m_sortByDesktop = m_showAllWindows && READ_MERGED_TASBKAR_SETTING(sortByDesktop); m_displayIconsNText = READ_MERGED_TASBKAR_SETTING(displayIconsNText); m_showOnlyIconified = READ_MERGED_TASBKAR_SETTING(showOnlyIconified); m_cycleWheel = READ_MERGED_TASBKAR_SETTING(cycleWheel); m_showTaskStates = READ_MERGED_TASBKAR_SETTING(showTaskStates); m_currentScreen = -1; // Show all screens or re-get our screen m_showOnlyCurrentScreen = (READ_MERGED_TASBKAR_SETTING(showCurrentScreenOnly) && TQApplication::desktop()->isVirtualDesktop() && TQApplication::desktop()->numScreens() > 1); // we need to watch geometry issues if we aren't showing windows when we // are paying attention to the current Xinerama screen // disconnect first in case we've been here before // to avoid multiple connections disconnect(TaskManager::the(), TQT_SIGNAL(windowChangedGeometry(Task::Ptr)), this, TQT_SLOT(windowChangedGeometry(Task::Ptr))); if (m_showOnlyCurrentScreen) { connect(TaskManager::the(), TQT_SIGNAL(windowChangedGeometry(Task::Ptr)), this, TQT_SLOT(windowChangedGeometry(Task::Ptr))); } TaskManager::the()->trackGeometry(m_showOnlyCurrentScreen); if (wasShowWindows != m_showAllWindows || wasSortByDesktop != m_sortByDesktop || wasDisplayIconsNText != m_displayIconsNText || wasCycleWheel != m_cycleWheel || wasShowOnlyIconified != m_showOnlyIconified || wasShowTaskStates != m_showTaskStates) { // relevant settings changed, update our task containers for (TaskContainer::Iterator it = containers.begin(); it != containers.end(); ++it) { (*it)->settingsChanged(); } } if (m_sortByAppPrev != READ_MERGED_TASBKAR_SETTING(sortByApp)) { m_sortByAppPrev = READ_MERGED_TASBKAR_SETTING(sortByApp); reSort(); } TaskManager::the()->setXCompositeEnabled(READ_MERGED_TASBKAR_SETTING(showThumbnails)); reLayoutEventually(); } void TaskBar::setOrientation( Orientation o ) { Panner::setOrientation( o ); reLayoutEventually(); } void TaskBar::moveEvent( TQMoveEvent* e ) { Panner::moveEvent(e); setViewportBackground(); } void TaskBar::resizeEvent( TQResizeEvent* e ) { if (m_showOnlyCurrentScreen) { TQPoint topLeft = mapToGlobal(this->geometry().topLeft()); if (m_currentScreen != TQApplication::desktop()->screenNumber(topLeft)) { // we have been moved to another screen! m_currentScreen = -1; reGroup(); } } Panner::resizeEvent(e); reLayoutEventually(); setViewportBackground(); } void TaskBar::add(Task::Ptr task) { if (!task || (m_showOnlyCurrentScreen && !TaskManager::isOnScreen(showScreen(), task->window()))) { return; } // try to group if (isGrouping) { for (TaskContainer::Iterator it = containers.begin(); it != containers.end(); ++it) { TaskContainer* c = *it; if (idMatch(task->classClass(), c->id())) { c->add(task); reLayoutEventually(); return; } } } // create new container TaskContainer *container = new TaskContainer(task, this, m_settingsObject, m_globalSettingsObject, viewport()); m_hiddenContainers.append(container); // even though there is a signal to listen to, we have to add this // immediately to ensure grouping doesn't break (primarily on startup) // we still add the container to m_hiddenContainers in case the event // loop gets re-entered here and something bizarre happens. call it // insurance =) showTaskContainer(container); } void TaskBar::add(Startup::Ptr startup) { if (!startup) { return; } for (TaskContainer::Iterator it = containers.begin(); it != containers.end(); ++it) { if ((*it)->contains(startup)) { return; } } // create new container TaskContainer *container = new TaskContainer(startup, frames, this, m_settingsObject, m_globalSettingsObject, viewport()); m_hiddenContainers.append(container); connect(container, TQT_SIGNAL(showMe(TaskContainer*)), this, TQT_SLOT(showTaskContainer(TaskContainer*))); } void TaskBar::reSort() { TaskContainer::List originalContainers = containers; TaskContainer::Iterator it = originalContainers.begin(); for (; it != originalContainers.end(); ++it) { removeChild(*it); } containers.clear(); it = originalContainers.begin(); for (; it != originalContainers.end(); ++it) { showTaskContainer(*it); } reLayoutEventually(); emit containerCountChanged(); } void TaskBar::showTaskContainer(TaskContainer* container) { TaskContainer::List::iterator it = m_hiddenContainers.find(container); if (it != m_hiddenContainers.end()) { m_hiddenContainers.erase(it); } if (container->isEmpty()) { return; } // try to place the container after one of the same app if (READ_MERGED_TASBKAR_SETTING(sortByApp)) { TaskContainer::Iterator it = containers.begin(); for (; it != containers.end(); ++it) { TaskContainer* c = *it; if (container->id().lower() == c->id().lower()) { // search for the last occurrence of this app for (; it != containers.end(); ++it) { c = *it; if (container->id().lower() != c->id().lower()) { break; } } break; } } // alphabetize containers it = containers.begin(); for (; it != containers.end(); ++it) { TaskContainer* c = *it; if (TQString::localeAwareCompare(container->id().lower(), c->id().lower()) < 0) { break; } } if (it != containers.end()) { containers.insert(it, container); } else { containers.append(container); } } else { containers.append(container); } addChild(container); reLayoutEventually(); emit containerCountChanged(); } void TaskBar::remove(Task::Ptr task, TaskContainer* container) { for (TaskContainer::Iterator it = m_hiddenContainers.begin(); it != m_hiddenContainers.end(); ++it) { if ((*it)->contains(task)) { (*it)->finish(); m_deletableContainers.append(*it); m_hiddenContainers.erase(it); break; } } if (!container) { for (TaskContainer::Iterator it = containers.begin(); it != containers.end(); ++it) { if ((*it)->contains(task)) { container = *it; break; } } if (!container) { return; } } container->remove(task); if (container->isEmpty()) { TaskContainer::List::iterator it = containers.find(container); if (it != containers.end()) { containers.erase(it); } removeChild(container); container->finish(); m_deletableContainers.append(container); reLayoutEventually(); emit containerCountChanged(); } else if (container->filteredTaskCount() < 1) { reLayoutEventually(); emit containerCountChanged(); } } void TaskBar::remove(Startup::Ptr startup, TaskContainer* container) { for (TaskContainer::Iterator it = m_hiddenContainers.begin(); it != m_hiddenContainers.end(); ++it) { if ((*it)->contains(startup)) { (*it)->remove(startup); if ((*it)->isEmpty()) { (*it)->finish(); m_deletableContainers.append(*it); m_hiddenContainers.erase(it); } break; } } if (!container) { for (TaskContainer::Iterator it = containers.begin(); it != containers.end(); ++it) { if ((*it)->contains(startup)) { container = *it; break; } } if (!container) { return; } } container->remove(startup); if (!container->isEmpty()) { return; } TaskContainer::List::iterator it = containers.find(container); if (it != containers.end()) { containers.erase(it); } // startup containers only ever contain that one item. so // just delete the poor bastard. container->finish(); m_deletableContainers.append(container); reLayoutEventually(); emit containerCountChanged(); } void TaskBar::desktopChanged(int desktop) { if (m_showAllWindows) { return; } m_relayoutTimer.stop(); m_ignoreUpdates = true; for (TaskContainer::Iterator it = containers.begin(); it != containers.end(); ++it) { (*it)->desktopChanged(desktop); } m_ignoreUpdates = false; reLayout(); emit containerCountChanged(); } void TaskBar::windowChanged(Task::Ptr task) { if (m_showOnlyCurrentScreen && !TaskManager::isOnScreen(showScreen(), task->window())) { return; // we don't care about this window } TaskContainer* container = 0; for (TaskContainer::List::const_iterator it = containers.constBegin(); it != containers.constEnd(); ++it) { TaskContainer* c = *it; if (c->contains(task)) { container = c; break; } } // if we don't have a container or we're showing only windows on this // desktop and the container is neither on the desktop nor currently visible // just skip it if (!container || (!m_showAllWindows && !container->onCurrentDesktop() && !container->isVisibleTo(this))) { return; } container->windowChanged(task); if (!m_showAllWindows || m_showOnlyIconified) { emit containerCountChanged(); } reLayoutEventually(); } void TaskBar::windowChangedGeometry(Task::Ptr task) { //TODO: this gets called every time a window's geom changes // when we are in "show only on the same Xinerama screen" // mode it would be Good(tm) to compress these events so this // gets run less often, but always when needed TaskContainer* container = 0; for (TaskContainer::Iterator it = containers.begin(); it != containers.end(); ++it) { TaskContainer* c = *it; if (c->contains(task)) { container = c; break; } } if ((!!container) == TaskManager::isOnScreen(showScreen(), task->window())) { // we have this window covered, so we don't need to do anything return; } if (container) { remove(task, container); } else { add(task); } } void TaskBar::reLayoutEventually() { m_relayoutTimer.stop(); if (!blocklayout && !m_ignoreUpdates) { m_relayoutTimer.start(25, true); } } void TaskBar::reLayout() { // Because TQPopupMenu::exec() creates its own event loop, deferred deletes // via TQObject::deleteLater() may be prematurely executed when a container's // popup menu is visible. // // To get around this, we collect the containers and delete them manually // when doing a relayout. (kling) if (!m_deletableContainers.isEmpty()) { TaskContainer::List::iterator it = m_deletableContainers.begin(); for (; it != m_deletableContainers.end(); ++it) delete *it; m_deletableContainers.clear(); } // filter task container list TaskContainer::List list = filteredContainers(); if (list.count() < 1) { resizeContents(contentsRect().width(), contentsRect().height()); return; } if (isGrouping != shouldGroup()) { reGroup(); return; } // sort container list by desktop if (m_sortByDesktop) { sortContainersByDesktop(list); } // needed because Panner doesn't know how big it's contents are so it's // up to us to initialize it. =( resizeContents(contentsRect().width(), contentsRect().height()); // number of rows simply depends on our height which is either the // minimum button height or the height of the font in use, whichever is // largest TQFontMetrics fm(TDEGlobalSettings::taskbarFont()); int minButtonHeight = fm.height() > READ_MERGED_TASBKAR_SETTING(minimumButtonHeight) ? fm.height() : READ_MERGED_TASBKAR_SETTING(minimumButtonHeight); // horizontal layout if (orientation() == Qt::Horizontal) { int bwidth=BUTTON_MIN_WIDTH; int rows = contentsRect().height() / minButtonHeight; if (rows<1) rows=1; // actual button height int bheight = contentsRect().height() / rows; if (bheight<1) // avoid zero devision later bheight=1; // buttons per row int bpr = static_cast<int>(ceil(static_cast<double>(list.count()) / rows)); // adjust content size if ( contentsRect().width() < bpr * BUTTON_MIN_WIDTH ) { resizeContents( bpr * BUTTON_MIN_WIDTH, contentsRect().height() ); } // maximum number of buttons per row int mbpr = contentsRect().width() / BUTTON_MIN_WIDTH; // expand button width if space permits and the taskbar is not in 'icons only' mode if (mbpr > bpr) { if (!showIcons() || showText()) bwidth = contentsRect().width() / bpr; int maxWidth = READ_MERGED_TASBKAR_SETTING(maximumButtonWidth); if (maxWidth > 0 && bwidth > maxWidth) { bwidth = maxWidth; } } // layout containers // for taskbars at the bottom, we need to ensure that the bottom // buttons touch the bottom of the screen. since we layout from // top to bottom this means seeing if we have any padding and // popping it on the top. this preserves Fitt's Law behaviour // for taskbars on the bottom int topPadding = 0; if (arrowType == UpArrow) { topPadding = contentsRect().height() % (rows * bheight); } int i = 0; bool reverseLayout = TQApplication::reverseLayout(); for (TaskContainer::Iterator it = list.begin(); it != list.end(); ++it, i++) { TaskContainer* c = *it; int row = i % rows; int x = ( i / rows ) * bwidth; if (reverseLayout) { x = contentsRect().width() - x - bwidth; } int y = (row * bheight) + topPadding; c->setArrowType(arrowType); if (childX(c) != x || childY(c) != y) moveChild(c, x, y); if (c->width() != bwidth || c->height() != bheight) c->resize( bwidth, bheight ); c->setBackground(); } } else // vertical layout { // Adjust min button height to keep gaps into account int minButtonHeightAdjusted=minButtonHeight+4; // adjust content size if (contentsRect().height() < (int)list.count() * minButtonHeightAdjusted) { resizeContents(contentsRect().width(), list.count() * minButtonHeightAdjusted); } // layout containers int i = 0; for (TaskContainer::Iterator it = list.begin(); it != list.end(); ++it) { TaskContainer* c = *it; c->setArrowType(arrowType); if (c->width() != contentsRect().width() || c->height() != minButtonHeightAdjusted) c->resize(contentsRect().width(), minButtonHeightAdjusted); if (childX(c) != 0 || childY(c) != (i * minButtonHeightAdjusted)) moveChild(c, 0, i * minButtonHeightAdjusted); c->setBackground(); i++; } } TQTimer::singleShot(100, this, TQT_SLOT(publishIconGeometry())); } void TaskBar::setViewportBackground() { const TQPixmap *bg = parentWidget()->backgroundPixmap(); if (bg) { TQPixmap pm(parentWidget()->size()); pm.fill(parentWidget(), pos() + viewport()->pos()); viewport()->setPaletteBackgroundPixmap(pm); viewport()->setBackgroundOrigin(WidgetOrigin); } else viewport()->setPaletteBackgroundColor(paletteBackgroundColor()); } void TaskBar::setBackground() { setViewportBackground(); TaskContainer::List list = filteredContainers(); for (TaskContainer::Iterator it = list.begin(); it != list.end(); ++it) { TaskContainer* c = *it; c->setBackground(); } } void TaskBar::setArrowType(TQt::ArrowType at) { if (arrowType == at) { return; } arrowType = at; for (TaskContainer::Iterator it = containers.begin(); it != containers.end(); ++it) { (*it)->setArrowType(arrowType); } } void TaskBar::publishIconGeometry() { TQPoint p = mapToGlobal(TQPoint(0,0)); // roundtrip, don't do that too often for (TaskContainer::Iterator it = containers.begin(); it != containers.end(); ++it) { (*it)->publishIconGeometry(p); } } void TaskBar::viewportMousePressEvent( TQMouseEvent* e ) { propagateMouseEvent( e ); } void TaskBar::viewportMouseReleaseEvent( TQMouseEvent* e ) { propagateMouseEvent( e ); } void TaskBar::viewportMouseDoubleClickEvent( TQMouseEvent* e ) { propagateMouseEvent( e ); } void TaskBar::viewportMouseMoveEvent( TQMouseEvent* e ) { propagateMouseEvent( e ); } void TaskBar::propagateMouseEvent( TQMouseEvent* e ) { if ( !isTopLevel() ) { TQMouseEvent me( e->type(), mapTo( topLevelWidget(), e->pos() ), e->globalPos(), e->button(), e->state() ); TQApplication::sendEvent( topLevelWidget(), &me ); } } bool TaskBar::idMatch( const TQString& id1, const TQString& id2 ) { if ( id1.isEmpty() || id2.isEmpty() ) return false; return id1.lower() == id2.lower(); } int TaskBar::containerCount() const { int i = 0; for (TaskContainer::List::const_iterator it = containers.constBegin(); it != containers.constEnd(); ++it) { if ((m_showAllWindows || (*it)->onCurrentDesktop()) && ((showScreen() == -1) || ((*it)->isOnScreen()))) { if (!(*it)->isHidden()) { i++; } } } return i; } int TaskBar::taskCount() const { int i = 0; for (TaskContainer::List::const_iterator it = containers.constBegin(); it != containers.constEnd(); ++it) { if ((m_showAllWindows || (*it)->onCurrentDesktop()) && ((showScreen() == -1) || ((*it)->isOnScreen()))) { if (!(*it)->isHidden()) { i += (*it)->filteredTaskCount(); } } } return i; } int TaskBar::maximumButtonsWithoutShrinking() const { TQFontMetrics fm(TDEGlobalSettings::taskbarFont()); int minButtonHeight = fm.height() > READ_MERGED_TASBKAR_SETTING(minimumButtonHeight) ? fm.height() : READ_MERGED_TASBKAR_SETTING(minimumButtonHeight); int rows = contentsRect().height() / minButtonHeight; if (rows < 1) { rows = 1; } if ( orientation() == Qt::Horizontal ) { // maxWidth of 0 means no max width, drop back to default int maxWidth = READ_MERGED_TASBKAR_SETTING(maximumButtonWidth); if (maxWidth == 0) { maxWidth = BUTTON_MAX_WIDTH; } // They squash a bit before they pop, hence the 2 return rows * (contentsRect().width() / maxWidth) + 2; } else { // Overlap slightly and ugly arrows appear, hence -1 return rows - 1; } } bool TaskBar::shouldGroup() const { return READ_MERGED_TASBKAR_SETTING(groupTasks) == m_settingsObject->GroupAlways || ((READ_MERGED_TASBKAR_SETTING(groupTasks) == m_settingsObject->GroupWhenFull && taskCount() > maximumButtonsWithoutShrinking())); } void TaskBar::reGroup() { isGrouping = shouldGroup(); blocklayout = true; TaskContainer::Iterator lastContainer = m_hiddenContainers.end(); for (TaskContainer::Iterator it = m_hiddenContainers.begin(); it != lastContainer; ++it) { (*it)->finish(); m_deletableContainers.append(*it); } m_hiddenContainers.clear(); for (TaskContainer::List::const_iterator it = containers.constBegin(); it != containers.constEnd(); ++it) { (*it)->finish(); m_deletableContainers.append(*it); } containers.clear(); Task::Dict tasks = TaskManager::the()->tasks(); Task::Dict::iterator lastTask = tasks.end(); for (Task::Dict::iterator it = tasks.begin(); it != lastTask; ++it) { Task::Ptr task = it.data(); if (showScreen() == -1 || task->isOnScreen(showScreen())) { add(task); } } Startup::List startups = TaskManager::the()->startups(); Startup::List::iterator itEnd = startups.end(); for (Startup::List::iterator sIt = startups.begin(); sIt != itEnd; ++sIt) { add(*sIt); } blocklayout = false; reLayoutEventually(); } TaskContainer::List TaskBar::filteredContainers() { // filter task container list TaskContainer::List list; for (TaskContainer::List::const_iterator it = containers.constBegin(); it != containers.constEnd(); ++it) { TaskContainer* c = *it; if ((m_showAllWindows || c->onCurrentDesktop()) && (!m_showOnlyIconified || c->isIconified()) && ((showScreen() == -1) || c->isOnScreen()) && (!c->isHidden())) { list.append(c); c->show(); } else { c->hide(); } } return list; } void TaskBar::activateNextTask(bool forward) { bool forcenext = false; TaskContainer::List list = filteredContainers(); // this is necessary here, because 'containers' is unsorted and // we want to iterate over the _shown_ task containers in a linear way if (m_sortByDesktop) { sortContainersByDesktop(list); } int numContainers = list.count(); TaskContainer::List::iterator it; for (int i = 0; i < numContainers; ++i) { it = forward ? list.at(i) : list.at(numContainers - i - 1); if (it != list.end() && (*it)->activateNextTask(forward, forcenext)) { return; } } if (forcenext) { // moving forward from the last, or backward from the first, loop around for (int i = 0; i < numContainers; ++i) { it = forward ? list.at(i) : list.at(numContainers - i - 1); if (it != list.end() && (*it)->activateNextTask(forward, forcenext)) { return; } } return; } forcenext = true; // select first for (int i = 0; i < numContainers; ++i) { it = forward ? list.at(i) : list.at(numContainers - i - 1); if (it == list.end()) { break; } TaskContainer* c = *it; if (m_sortByDesktop) { if (forward ? c->desktop() < TaskManager::the()->currentDesktop() : c->desktop() > TaskManager::the()->currentDesktop()) { continue; } } if (c->activateNextTask(forward, forcenext)) { return; } } } void TaskBar::wheelEvent(TQWheelEvent* e) { if(READ_MERGED_TASBKAR_SETTING(cycleWheel)) { if (e->delta() > 0) { // scroll away from user, previous task activateNextTask(false); } else { // scroll towards user, next task activateNextTask(true); } } } void TaskBar::slotActivateNextTask() { activateNextTask( true ); } void TaskBar::slotActivatePreviousTask() { activateNextTask( false ); } void TaskBar::slotSettingsChanged( int category ) { if( category == (int) TDEApplication::SETTINGS_SHORTCUTS ) { keys->readSettings(); keys->updateConnections(); } } int TaskBar::showScreen() const { if (m_showOnlyCurrentScreen && m_currentScreen == -1) { const_cast<TaskBar*>(this)->m_currentScreen = TQApplication::desktop()->screenNumber(mapToGlobal(this->geometry().topLeft())); } return m_currentScreen; } TQImage* TaskBar::blendGradient(const TQSize& size) { if (m_blendGradient.isNull() || m_blendGradient.size() != size) { TQPixmap bgpm(size); TQPainter bgp(&bgpm); bgpm.fill(black); if (TQApplication::reverseLayout()) { TQImage gradient = KImageEffect::gradient( TQSize(30, size.height()), TQColor(255,255,255), TQColor(0,0,0), KImageEffect::HorizontalGradient); bgp.drawImage(0, 0, gradient); } else { TQImage gradient = KImageEffect::gradient( TQSize(30, size.height()), TQColor(0,0,0), TQColor(255,255,255), KImageEffect::HorizontalGradient); bgp.drawImage(size.width() - 30, 0, gradient); } m_blendGradient = bgpm.convertToImage(); } return &m_blendGradient; } void TaskBar::sortContainersByDesktop(TaskContainer::List& list) { typedef TQValueVector<QPair<int, QPair<int, TaskContainer*> > > SortVector; SortVector sorted; sorted.resize(list.count()); int i = 0; TaskContainer::List::ConstIterator lastUnsorted(list.constEnd()); for (TaskContainer::List::ConstIterator it = list.constBegin(); it != lastUnsorted; ++it) { sorted[i] = qMakePair((*it)->desktop(), qMakePair(i, *it)); ++i; } qHeapSort(sorted); list.clear(); SortVector::const_iterator lastSorted(sorted.constEnd()); for (SortVector::const_iterator it = sorted.constBegin(); it != lastSorted; ++it) { list.append((*it).second.second); } } TaskMoveDestination::TaskMoveDestination TaskBar::taskMoveCapabilities(TaskContainer* movingContainer) { TaskMoveDestination::TaskMoveDestination ret = TaskMoveDestination::Null; bool before = false; bool after = false; bool movingFound = false; if (movingContainer) { // Check to see if there are any visible containers before or after the movingContainer TaskContainer::Iterator it = containers.begin(); for (; it != containers.end(); ++it) { TaskContainer* c = *it; if (!c->isVisibleTo(this)) { continue; } if (c == movingContainer) { movingFound = true; } else { if (movingFound) { after = true; } else { before = true; } } } if (before) { ret = ret | TaskMoveDestination::Left; } if (after) { ret = ret | TaskMoveDestination::Right; } } return ret; } int TaskBar::taskMoveHandler(TaskMoveDestination::TaskMoveDestination dest, Task::List taskList, const TQPoint pos) { TaskContainer* movingContainer = NULL; TaskContainer* destContainer = NULL; bool movingRight = true; TaskContainer::Iterator it = containers.begin(); for (; it != containers.end(); ++it) { TaskContainer* c = *it; if (!c->isVisibleTo(this)) { continue; } if (c->taskList() == taskList) { movingContainer = c; break; } } if (movingContainer) { if (dest == TaskMoveDestination::Position) { // Find the best place for the container to go... it = containers.begin(); for (; it != containers.end(); ++it) { TaskContainer* c = *it; if (!c->isVisibleTo(this)) { continue; } TQPoint containerPos = c->pos(); TQSize containerSize = c->size(); TQRect containerRect(containerPos.x(), containerPos.y(), containerSize.width(), containerSize.height()); if (containerRect.contains(pos)) { destContainer = c; // Figure out if the mobile container is moving towards the end of the container list (i.e. right or down) for (; it != containers.end(); ++it) { if (movingContainer == (*it)) { movingRight = false; } } break; } } } else if (dest == TaskMoveDestination::Beginning) { // Move to beginning it = containers.begin(); while ((it != containers.end()) && (!(*it)->isVisibleTo(this))) { it++; } if (it == containers.end()) { return false; } destContainer = *it; movingRight = false; } else if (dest == TaskMoveDestination::Left) { // Move left it = containers.begin(); while ((it != containers.end()) && (!(*it)->isVisibleTo(this))) { it++; } if (it == containers.end()) { return false; } TaskContainer* prev = *it; destContainer = prev; for (; it != containers.end(); ++it) { TaskContainer* c = *it; if (!c->isVisibleTo(this)) { continue; } if (movingContainer == c) { destContainer = prev; break; } prev = c; } movingRight = false; } else if (dest == TaskMoveDestination::Right) { // Move right it = containers.begin(); destContainer = NULL; for (; it != containers.end(); ++it) { TaskContainer* c = *it; if (!c->isVisibleTo(this)) { continue; } if (movingContainer == c) { if (it != containers.end()) { it++; while ((it != containers.end()) && (!(*it)->isVisibleTo(this))) { it++; } } if ((it != containers.end()) && ((*it)->isVisibleTo(this))) { destContainer = *it; } break; } } movingRight = true; } else if (dest == TaskMoveDestination::End) { // Move to end destContainer = NULL; movingRight = true; } if (destContainer == movingContainer) { return false; } removeChild(movingContainer); containers.remove(movingContainer); if (destContainer) { it = containers.find(destContainer); if ((it != containers.end()) && (movingRight)) { it++; while ((it != containers.end()) && (!(*it)->isVisibleTo(this))) { it++; } } if ((it != containers.end()) && ((*it)->isVisibleTo(this))) { containers.insert(it, movingContainer); } else { containers.append(movingContainer); } } else { containers.append(movingContainer); } addChild(movingContainer); reLayoutEventually(); emit containerCountChanged(); return true; } return false; }