openbox/src/Workspace.cc

711 lines
21 KiB
C++
Raw Normal View History

// -*- mode: C++; indent-tabs-mode: nil; c-basic-offset: 2; -*-
// Workspace.cc for Blackbox - an X11 Window manager
// Copyright (c) 2001 - 2002 Sean 'Shaleh' Perry <shaleh@debian.org>
2002-04-11 03:20:38 +00:00
// Copyright (c) 1997 - 2000 Brad Hughes (bhughes@tcac.net)
//
// 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 OR COPYRIGHT HOLDERS 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.
#ifdef HAVE_CONFIG_H
# include "../config.h"
#endif // HAVE_CONFIG_H
extern "C" {
#include <X11/Xlib.h>
#include <X11/Xatom.h>
#ifdef HAVE_STDIO_H
# include <stdio.h>
#endif // HAVE_STDIO_H
#ifdef HAVE_STRING_H
# include <string.h>
#endif // HAVE_STRING_H
}
2002-06-01 17:54:32 +00:00
#include <assert.h>
#include <functional>
#include <string>
2002-04-11 03:20:38 +00:00
using std::string;
2002-04-11 03:20:38 +00:00
#include "i18n.hh"
#include "blackbox.hh"
#include "Clientmenu.hh"
#include "Netizen.hh"
#include "Screen.hh"
#include "Toolbar.hh"
#include "Util.hh"
#include "Window.hh"
#include "Workspace.hh"
#include "Windowmenu.hh"
#include "XAtom.hh"
2002-04-11 03:20:38 +00:00
Workspace::Workspace(BScreen *scrn, unsigned int i) {
screen = scrn;
xatom = screen->getBlackbox()->getXAtom();
2002-04-11 03:20:38 +00:00
cascade_x = cascade_y = 32;
2002-04-11 03:20:38 +00:00
id = i;
2002-04-11 03:20:38 +00:00
clientmenu = new Clientmenu(this);
lastfocus = (BlackboxWindow *) 0;
readName();
2002-04-11 03:20:38 +00:00
}
void Workspace::addWindow(BlackboxWindow *w, bool place) {
assert(w != 0);
2002-04-11 03:20:38 +00:00
if (place) placeWindow(w);
2002-04-11 03:20:38 +00:00
w->setWorkspace(id);
w->setWindowNumber(windowList.size());
2002-04-11 03:20:38 +00:00
stackingList.push_front(w);
windowList.push_back(w);
2002-04-11 03:20:38 +00:00
clientmenu->insert(w->getTitle());
2002-04-11 03:20:38 +00:00
clientmenu->update();
screen->updateNetizenWindowAdd(w->getClientWindow(), id);
2002-04-11 03:20:38 +00:00
raiseWindow(w);
}
unsigned int Workspace::removeWindow(BlackboxWindow *w) {
assert(w != 0);
2002-04-11 03:20:38 +00:00
stackingList.remove(w);
2002-04-11 03:20:38 +00:00
2002-06-21 01:06:29 +00:00
// pass focus to the next appropriate window
if ((w->isFocused() || w == lastfocus) &&
! screen->getBlackbox()->doShutdown()) {
2002-07-14 18:45:46 +00:00
focusFallback(w);
// if the window is sticky, then it needs to be removed on all other
// workspaces too!
if (w->isStuck()) {
for (unsigned int i = 0; i < screen->getWorkspaceCount(); ++i)
if (i != id)
screen->getWorkspace(i)->focusFallback(w);
2002-06-21 01:06:29 +00:00
}
}
windowList.remove(w);
2002-04-11 03:20:38 +00:00
clientmenu->remove(w->getWindowNumber());
clientmenu->update();
screen->updateNetizenWindowDel(w->getClientWindow());
2002-04-11 03:20:38 +00:00
BlackboxWindowList::iterator it = windowList.begin();
const BlackboxWindowList::iterator end = windowList.end();
unsigned int i = 0;
for (; it != end; ++it, ++i)
(*it)->setWindowNumber(i);
2002-04-11 03:20:38 +00:00
if (i == 0)
cascade_x = cascade_y = 32;
2002-04-11 03:20:38 +00:00
return i;
}
2002-07-10 22:24:48 +00:00
void Workspace::focusFallback(const BlackboxWindow *old_window) {
BlackboxWindow *newfocus = 0;
2002-07-14 18:45:46 +00:00
if (id == screen->getCurrentWorkspaceID()) {
// The window is on the visible workspace.
2002-07-10 22:24:48 +00:00
2002-07-14 18:45:46 +00:00
// if it's a transient, then try to focus its parent
if (old_window && old_window->isTransient()) {
newfocus = old_window->getTransientFor();
if (! newfocus ||
newfocus->isIconic() || // do not focus icons
newfocus->getWorkspaceNumber() != id || // or other workspaces
! newfocus->setInputFocus())
newfocus = 0;
}
2002-07-10 22:24:48 +00:00
2002-07-14 18:45:46 +00:00
if (! newfocus) {
BlackboxWindowList::iterator it = stackingList.begin(),
end = stackingList.end();
for (; it != end; ++it) {
BlackboxWindow *tmp = *it;
if (tmp && tmp->setInputFocus()) {
// we found our new focus target
newfocus = tmp;
break;
}
2002-07-10 22:24:48 +00:00
}
}
2002-07-14 18:45:46 +00:00
screen->getBlackbox()->setFocusedWindow(newfocus);
} else {
// The window is not on the visible workspace.
if (old_window && lastfocus == old_window) {
// The window was the last-focus target, so we need to replace it.
BlackboxWindow *win = (BlackboxWindow*) 0;
if (! stackingList.empty())
win = stackingList.front();
setLastFocusedWindow(win);
}
}
2002-07-10 22:24:48 +00:00
}
2002-04-11 03:20:38 +00:00
void Workspace::showAll(void) {
std::for_each(stackingList.begin(), stackingList.end(),
std::mem_fun(&BlackboxWindow::show));
2002-04-11 03:20:38 +00:00
}
void Workspace::hideAll(void) {
// withdraw in reverse order to minimize the number of Expose events
2002-04-11 03:20:38 +00:00
BlackboxWindowList lst(stackingList.rbegin(), stackingList.rend());
2002-04-11 03:20:38 +00:00
BlackboxWindowList::iterator it = lst.begin();
const BlackboxWindowList::iterator end = lst.end();
for (; it != end; ++it) {
BlackboxWindow *bw = *it;
if (! bw->isStuck())
bw->withdraw();
}
2002-04-11 03:20:38 +00:00
}
void Workspace::removeAll(void) {
while (! windowList.empty())
windowList.front()->iconify();
}
2002-04-11 03:20:38 +00:00
/*
* returns the number of transients for win, plus the number of transients
* associated with each transient of win
*/
static int countTransients(const BlackboxWindow * const win) {
int ret = win->getTransients().size();
if (ret > 0) {
BlackboxWindowList::const_iterator it, end = win->getTransients().end();
for (it = win->getTransients().begin(); it != end; ++it) {
ret += countTransients(*it);
}
}
return ret;
}
/*
* puts the transients of win into the stack. windows are stacked above
* the window before it in the stackvector being iterated, meaning
* stack[0] is on bottom, stack[1] is above stack[0], stack[2] is above
* stack[1], etc...
*/
void Workspace::raiseTransients(const BlackboxWindow * const win,
StackVector::iterator &stack) {
if (win->getTransients().size() == 0) return; // nothing to do
// put win's transients in the stack
BlackboxWindowList::const_iterator it, end = win->getTransients().end();
for (it = win->getTransients().begin(); it != end; ++it) {
*stack++ = (*it)->getFrameWindow();
screen->updateNetizenWindowRaise((*it)->getClientWindow());
if (! (*it)->isIconic()) {
Workspace *wkspc = screen->getWorkspace((*it)->getWorkspaceNumber());
wkspc->stackingList.remove((*it));
wkspc->stackingList.push_front((*it));
}
}
2002-04-11 03:20:38 +00:00
// put transients of win's transients in the stack
for (it = win->getTransients().begin(); it != end; ++it) {
raiseTransients(*it, stack);
2002-04-11 03:20:38 +00:00
}
}
2002-04-11 03:20:38 +00:00
void Workspace::lowerTransients(const BlackboxWindow * const win,
StackVector::iterator &stack) {
if (win->getTransients().size() == 0) return; // nothing to do
2002-04-11 03:20:38 +00:00
// put transients of win's transients in the stack
BlackboxWindowList::const_reverse_iterator it,
end = win->getTransients().rend();
for (it = win->getTransients().rbegin(); it != end; ++it) {
lowerTransients(*it, stack);
}
2002-04-11 03:20:38 +00:00
// put win's transients in the stack
for (it = win->getTransients().rbegin(); it != end; ++it) {
*stack++ = (*it)->getFrameWindow();
screen->updateNetizenWindowLower((*it)->getClientWindow());
2002-04-11 03:20:38 +00:00
if (! (*it)->isIconic()) {
Workspace *wkspc = screen->getWorkspace((*it)->getWorkspaceNumber());
wkspc->stackingList.remove((*it));
wkspc->stackingList.push_back((*it));
}
2002-04-11 03:20:38 +00:00
}
}
void Workspace::raiseWindow(BlackboxWindow *w) {
BlackboxWindow *win = w;
// walk up the transient_for's to the window that is not a transient
while (win->isTransient()) {
if (! win->getTransientFor()) break;
win = win->getTransientFor();
}
2002-04-11 03:20:38 +00:00
// get the total window count (win and all transients)
unsigned int i = 1 + countTransients(win);
2002-04-11 03:20:38 +00:00
// stack the window with all transients above
StackVector stack_vector(i);
StackVector::iterator stack = stack_vector.begin();
2002-04-11 03:20:38 +00:00
*(stack++) = win->getFrameWindow();
screen->updateNetizenWindowRaise(win->getClientWindow());
if (! win->isIconic()) {
Workspace *wkspc = screen->getWorkspace(win->getWorkspaceNumber());
wkspc->stackingList.remove(win);
wkspc->stackingList.push_front(win);
2002-04-11 03:20:38 +00:00
}
raiseTransients(win, stack);
2002-04-11 03:20:38 +00:00
screen->raiseWindows(&stack_vector[0], stack_vector.size());
}
2002-04-11 03:20:38 +00:00
void Workspace::lowerWindow(BlackboxWindow *w) {
BlackboxWindow *win = w;
2002-04-11 03:20:38 +00:00
// walk up the transient_for's to the window that is not a transient
while (win->isTransient()) {
if (! win->getTransientFor()) break;
2002-04-11 03:20:38 +00:00
win = win->getTransientFor();
}
// get the total window count (win and all transients)
unsigned int i = 1 + countTransients(win);
// stack the window with all transients above
StackVector stack_vector(i);
StackVector::iterator stack = stack_vector.begin();
2002-04-11 03:20:38 +00:00
lowerTransients(win, stack);
2002-04-11 03:20:38 +00:00
*(stack++) = win->getFrameWindow();
screen->updateNetizenWindowLower(win->getClientWindow());
if (! win->isIconic()) {
Workspace *wkspc = screen->getWorkspace(win->getWorkspaceNumber());
wkspc->stackingList.remove(win);
wkspc->stackingList.push_back(win);
}
2002-04-11 03:20:38 +00:00
screen->lowerWindows(&stack_vector[0], stack_vector.size());
2002-04-11 03:20:38 +00:00
}
void Workspace::reconfigure(void) {
clientmenu->reconfigure();
std::for_each(windowList.begin(), windowList.end(),
std::mem_fun(&BlackboxWindow::reconfigure));
}
2002-04-11 03:20:38 +00:00
BlackboxWindow *Workspace::getWindow(unsigned int index) {
if (index < windowList.size()) {
BlackboxWindowList::iterator it = windowList.begin();
for(; index > 0; --index, ++it); /* increment to index */
return *it;
}
return 0;
2002-04-11 03:20:38 +00:00
}
BlackboxWindow*
Workspace::getNextWindowInList(BlackboxWindow *w) {
BlackboxWindowList::iterator it = std::find(windowList.begin(),
windowList.end(),
w);
assert(it != windowList.end()); // window must be in list
++it; // next window
if (it == windowList.end())
return windowList.front(); // if we walked off the end, wrap around
return *it;
2002-04-11 03:20:38 +00:00
}
BlackboxWindow* Workspace::getPrevWindowInList(BlackboxWindow *w) {
BlackboxWindowList::iterator it = std::find(windowList.begin(),
windowList.end(),
w);
assert(it != windowList.end()); // window must be in list
if (it == windowList.begin())
return windowList.back(); // if we walked of the front, wrap around
return *(--it);
2002-04-11 03:20:38 +00:00
}
BlackboxWindow* Workspace::getTopWindowOnStack(void) const {
return stackingList.front();
2002-04-11 03:20:38 +00:00
}
void Workspace::sendWindowList(Netizen &n) {
BlackboxWindowList::iterator it = windowList.begin(),
end = windowList.end();
for(; it != end; ++it)
n.sendWindowAdd((*it)->getClientWindow(), getID());
}
unsigned int Workspace::getCount(void) const {
return windowList.size();
}
void Workspace::appendStackOrder(BlackboxWindowList &stack_order) const {
BlackboxWindowList::const_reverse_iterator it = stackingList.rbegin();
const BlackboxWindowList::const_reverse_iterator end = stackingList.rend();
for (; it != end; ++it)
stack_order.push_back(*it);
}
bool Workspace::isCurrent(void) const {
return (id == screen->getCurrentWorkspaceID());
2002-04-11 03:20:38 +00:00
}
bool Workspace::isLastWindow(const BlackboxWindow* const w) const {
return (w == windowList.back());
}
void Workspace::setCurrent(void) {
screen->changeWorkspaceID(id);
}
2002-04-11 03:20:38 +00:00
void Workspace::readName(void) {
XAtom::StringVect namesList;
unsigned long numnames = id + 1;
// attempt to get from the _NET_WM_DESKTOP_NAMES property
if (xatom->getValue(screen->getRootWindow(), XAtom::net_desktop_names,
XAtom::utf8, numnames, namesList) &&
namesList.size() > id) {
name = namesList[id];
clientmenu->setLabel(name);
clientmenu->update();
2002-04-11 03:20:38 +00:00
} else {
/*
Use a default name. This doesn't actually change the class. That will
happen after the setName changes the root property, and that change
makes its way back to this function.
*/
string tmp =i18n(WorkspaceSet, WorkspaceDefaultNameFormat,
"Workspace %d");
assert(tmp.length() < 32);
char default_name[32];
sprintf(default_name, tmp.c_str(), id + 1);
setName(default_name); // save this into the _NET_WM_DESKTOP_NAMES property
}
}
void Workspace::setName(const string& new_name) {
// set the _NET_WM_DESKTOP_NAMES property with the new name
XAtom::StringVect namesList;
unsigned long numnames = (unsigned) -1;
if (xatom->getValue(screen->getRootWindow(), XAtom::net_desktop_names,
XAtom::utf8, numnames, namesList) &&
namesList.size() > id)
namesList[id] = new_name;
else
namesList.push_back(new_name);
xatom->setValue(screen->getRootWindow(), XAtom::net_desktop_names,
XAtom::utf8, namesList);
2002-04-11 03:20:38 +00:00
}
/*
* Calculate free space available for window placement.
*/
typedef std::vector<Rect> rectList;
2002-04-11 03:20:38 +00:00
static rectList calcSpace(const Rect &win, const rectList &spaces) {
Rect isect, extra;
rectList result;
rectList::const_iterator siter, end = spaces.end();
for (siter = spaces.begin(); siter != end; ++siter) {
const Rect &curr = *siter;
if(! win.intersects(curr)) {
result.push_back(curr);
continue;
}
/* Use an intersection of win and curr to determine the space around
* curr that we can use.
*
* NOTE: the spaces calculated can overlap.
*/
isect = curr & win;
// left
extra.setCoords(curr.left(), curr.top(),
isect.left() - 1, curr.bottom());
if (extra.valid()) result.push_back(extra);
// top
extra.setCoords(curr.left(), curr.top(),
curr.right(), isect.top() - 1);
if (extra.valid()) result.push_back(extra);
// right
extra.setCoords(isect.right() + 1, curr.top(),
curr.right(), curr.bottom());
if (extra.valid()) result.push_back(extra);
// bottom
extra.setCoords(curr.left(), isect.bottom() + 1,
curr.right(), curr.bottom());
if (extra.valid()) result.push_back(extra);
}
return result;
}
static bool rowRLBT(const Rect &first, const Rect &second) {
if (first.bottom() == second.bottom())
return first.right() > second.right();
return first.bottom() > second.bottom();
}
static bool rowRLTB(const Rect &first, const Rect &second) {
if (first.y() == second.y())
return first.right() > second.right();
return first.y() < second.y();
}
static bool rowLRBT(const Rect &first, const Rect &second) {
if (first.bottom() == second.bottom())
return first.x() < second.x();
return first.bottom() > second.bottom();
}
static bool rowLRTB(const Rect &first, const Rect &second) {
if (first.y() == second.y())
return first.x() < second.x();
return first.y() < second.y();
}
static bool colLRTB(const Rect &first, const Rect &second) {
if (first.x() == second.x())
return first.y() < second.y();
return first.x() < second.x();
}
static bool colLRBT(const Rect &first, const Rect &second) {
if (first.x() == second.x())
return first.bottom() > second.bottom();
return first.x() < second.x();
}
static bool colRLTB(const Rect &first, const Rect &second) {
if (first.right() == second.right())
return first.y() < second.y();
return first.right() > second.right();
}
static bool colRLBT(const Rect &first, const Rect &second) {
if (first.right() == second.right())
return first.bottom() > second.bottom();
return first.right() > second.right();
}
bool Workspace::smartPlacement(Rect& win, const Rect& availableArea) {
rectList spaces;
spaces.push_back(availableArea); //initially the entire screen is free
//Find Free Spaces
2002-05-30 04:35:22 +00:00
BlackboxWindowList::const_iterator wit = windowList.begin(),
end = windowList.end();
Rect tmp;
for (; wit != end; ++wit) {
const BlackboxWindow* const curr = *wit;
2002-05-30 04:35:22 +00:00
if (curr->isShaded() && screen->getPlaceIgnoreShaded()) continue;
if (curr->isMaximizedFull() && screen->getPlaceIgnoreMaximized()) continue;
2002-05-30 04:35:22 +00:00
tmp.setRect(curr->frameRect().x(), curr->frameRect().y(),
curr->frameRect().width() + screen->getBorderWidth(),
curr->frameRect().height() + screen->getBorderWidth());
spaces = calcSpace(tmp, spaces);
}
if (screen->getPlacementPolicy() == BScreen::RowSmartPlacement) {
if(screen->getRowPlacementDirection() == BScreen::LeftRight) {
if(screen->getColPlacementDirection() == BScreen::TopBottom)
2002-06-01 17:54:32 +00:00
std::sort(spaces.begin(), spaces.end(), rowLRTB);
else
2002-06-01 17:54:32 +00:00
std::sort(spaces.begin(), spaces.end(), rowLRBT);
} else {
if(screen->getColPlacementDirection() == BScreen::TopBottom)
2002-06-01 17:54:32 +00:00
std::sort(spaces.begin(), spaces.end(), rowRLTB);
else
2002-06-01 17:54:32 +00:00
std::sort(spaces.begin(), spaces.end(), rowRLBT);
}
} else {
if(screen->getColPlacementDirection() == BScreen::TopBottom) {
if(screen->getRowPlacementDirection() == BScreen::LeftRight)
2002-06-01 17:54:32 +00:00
std::sort(spaces.begin(), spaces.end(), colLRTB);
else
2002-06-01 17:54:32 +00:00
std::sort(spaces.begin(), spaces.end(), colRLTB);
} else {
if(screen->getRowPlacementDirection() == BScreen::LeftRight)
2002-06-01 17:54:32 +00:00
std::sort(spaces.begin(), spaces.end(), colLRBT);
else
2002-06-01 17:54:32 +00:00
std::sort(spaces.begin(), spaces.end(), colRLBT);
}
}
rectList::const_iterator sit = spaces.begin(), spaces_end = spaces.end();
for(; sit != spaces_end; ++sit) {
if (sit->width() >= win.width() && sit->height() >= win.height())
break;
}
if (sit == spaces_end)
return False;
//set new position based on the empty space found
const Rect& where = *sit;
win.setX(where.x());
win.setY(where.y());
// adjust the location() based on left/right and top/bottom placement
if (screen->getPlacementPolicy() == BScreen::RowSmartPlacement) {
if (screen->getRowPlacementDirection() == BScreen::RightLeft)
win.setX(where.right() - win.width());
if (screen->getColPlacementDirection() == BScreen::BottomTop)
win.setY(where.bottom() - win.height());
} else {
if (screen->getColPlacementDirection() == BScreen::BottomTop)
win.setY(win.y() + where.height() - win.height());
if (screen->getRowPlacementDirection() == BScreen::RightLeft)
win.setX(win.x() + where.width() - win.width());
}
return True;
}
bool Workspace::underMousePlacement(Rect &win, const Rect &availableArea) {
int x, y, rx, ry;
Window c, r;
unsigned int m;
XQueryPointer(screen->getBlackbox()->getXDisplay(), screen->getRootWindow(),
&r, &c, &rx, &ry, &x, &y, &m);
x = rx - win.width() / 2;
y = ry - win.height() / 2;
if (x < availableArea.x())
x = availableArea.x();
if (y < availableArea.y())
y = availableArea.y();
if (x + win.width() > availableArea.x() + availableArea.width())
x = availableArea.x() + availableArea.width() - win.width();
if (y + win.height() > availableArea.y() + availableArea.height())
y = availableArea.y() + availableArea.height() - win.height();
win.setX(x);
win.setY(y);
return True;
}
bool Workspace::cascadePlacement(Rect &win, const Rect &availableArea) {
if ((cascade_x > static_cast<signed>(availableArea.width() / 2)) ||
(cascade_y > static_cast<signed>(availableArea.height() / 2)))
cascade_x = cascade_y = 32;
if (cascade_x == 32) {
cascade_x += availableArea.x();
cascade_y += availableArea.y();
}
win.setPos(cascade_x, cascade_y);
return True;
}
void Workspace::placeWindow(BlackboxWindow *win) {
Rect availableArea(screen->availableArea()),
new_win(availableArea.x(), availableArea.y(),
win->frameRect().width(), win->frameRect().height());
bool placed = False;
switch (screen->getPlacementPolicy()) {
case BScreen::RowSmartPlacement:
case BScreen::ColSmartPlacement:
placed = smartPlacement(new_win, availableArea);
2002-04-28 00:11:19 +00:00
break;
case BScreen::UnderMousePlacement:
placed = underMousePlacement(new_win, availableArea);
default:
break; // handled below
2002-04-11 03:20:38 +00:00
} // switch
if (placed == False) {
cascadePlacement(new_win, availableArea);
cascade_x += win->getTitleHeight() + (screen->getBorderWidth() * 2);
cascade_y += win->getTitleHeight() + (screen->getBorderWidth() * 2);
}
if (new_win.right() > availableArea.right())
new_win.setX(availableArea.left());
if (new_win.bottom() > availableArea.bottom())
new_win.setY(availableArea.top());
win->configure(new_win.x(), new_win.y(), new_win.width(), new_win.height());
}