2002-05-25 00:29:44 +00:00
|
|
|
// -*- 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
|
|
|
|
|
2002-05-25 00:29:44 +00:00
|
|
|
extern "C" {
|
|
|
|
#include <X11/Xlib.h>
|
|
|
|
#include <X11/Xatom.h>
|
|
|
|
|
2002-05-11 05:33:49 +00:00
|
|
|
#ifdef HAVE_STDIO_H
|
|
|
|
# include <stdio.h>
|
|
|
|
#endif // HAVE_STDIO_H
|
|
|
|
|
2002-05-25 00:29:44 +00:00
|
|
|
#ifdef HAVE_STRING_H
|
2002-05-11 05:33:49 +00:00
|
|
|
# include <string.h>
|
|
|
|
#endif // HAVE_STRING_H
|
2002-05-25 00:29:44 +00:00
|
|
|
}
|
2002-05-11 05:33:49 +00:00
|
|
|
|
2002-06-01 17:54:32 +00:00
|
|
|
#include <assert.h>
|
|
|
|
|
2002-05-25 00:29:44 +00:00
|
|
|
#include <functional>
|
|
|
|
#include <string>
|
2002-04-11 03:20:38 +00:00
|
|
|
|
2002-05-25 00:29:44 +00:00
|
|
|
using std::string;
|
2002-04-11 03:20:38 +00:00
|
|
|
|
2002-05-25 00:29:44 +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"
|
2002-06-28 01:44:47 +00:00
|
|
|
#include "XAtom.hh"
|
2002-04-11 03:20:38 +00:00
|
|
|
|
|
|
|
|
2002-05-25 00:29:44 +00:00
|
|
|
Workspace::Workspace(BScreen *scrn, unsigned int i) {
|
|
|
|
screen = scrn;
|
2002-06-28 01:44:47 +00:00
|
|
|
xatom = screen->getBlackbox()->getXAtom();
|
2002-04-11 03:20:38 +00:00
|
|
|
|
2002-05-25 00:29:44 +00:00
|
|
|
cascade_x = cascade_y = 32;
|
2002-04-11 03:20:38 +00:00
|
|
|
|
2002-05-25 00:29:44 +00:00
|
|
|
id = i;
|
2002-04-11 03:20:38 +00:00
|
|
|
|
2002-05-25 00:29:44 +00:00
|
|
|
clientmenu = new Clientmenu(this);
|
|
|
|
|
|
|
|
lastfocus = (BlackboxWindow *) 0;
|
|
|
|
|
2002-06-28 01:44:47 +00:00
|
|
|
setName("");
|
2002-04-11 03:20:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2002-05-25 00:29:44 +00:00
|
|
|
void Workspace::addWindow(BlackboxWindow *w, bool place) {
|
|
|
|
assert(w != 0);
|
2002-04-11 03:20:38 +00:00
|
|
|
|
2002-05-25 00:29:44 +00:00
|
|
|
if (place) placeWindow(w);
|
2002-05-14 07:53:22 +00:00
|
|
|
|
2002-04-11 03:20:38 +00:00
|
|
|
w->setWorkspace(id);
|
2002-05-25 00:29:44 +00:00
|
|
|
w->setWindowNumber(windowList.size());
|
2002-04-11 03:20:38 +00:00
|
|
|
|
2002-05-25 00:29:44 +00:00
|
|
|
stackingList.push_front(w);
|
|
|
|
windowList.push_back(w);
|
2002-04-11 03:20:38 +00:00
|
|
|
|
2002-05-25 00:29:44 +00:00
|
|
|
clientmenu->insert(w->getTitle());
|
2002-04-11 03:20:38 +00:00
|
|
|
clientmenu->update();
|
|
|
|
|
2002-05-25 00:29:44 +00:00
|
|
|
screen->updateNetizenWindowAdd(w->getClientWindow(), id);
|
2002-04-11 03:20:38 +00:00
|
|
|
|
|
|
|
raiseWindow(w);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2002-05-25 00:29:44 +00:00
|
|
|
unsigned int Workspace::removeWindow(BlackboxWindow *w) {
|
|
|
|
assert(w != 0);
|
2002-04-11 03:20:38 +00:00
|
|
|
|
2002-05-25 00:29:44 +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()) {
|
|
|
|
if (id == screen->getCurrentWorkspaceID()) {
|
2002-07-10 22:24:48 +00:00
|
|
|
// The window is on the visible workspace
|
|
|
|
focusFallback(w);
|
|
|
|
} else {
|
|
|
|
// The window is not on the visible workspace.
|
|
|
|
if (lastfocus == w) {
|
|
|
|
// The window was the last-focus target, so we need to replace it.
|
|
|
|
setLastFocusedWindow(stackingList.front());
|
|
|
|
}
|
|
|
|
// if the window focused on the current workspace, then reapply that
|
|
|
|
// workspace's focus too
|
|
|
|
if (w->isFocused())
|
|
|
|
screen->getCurrentWorkspace()->focusFallback(w);
|
2002-06-21 01:06:29 +00:00
|
|
|
}
|
|
|
|
}
|
2002-05-25 00:29:44 +00:00
|
|
|
|
|
|
|
windowList.remove(w);
|
2002-04-11 03:20:38 +00:00
|
|
|
clientmenu->remove(w->getWindowNumber());
|
|
|
|
clientmenu->update();
|
|
|
|
|
2002-05-25 00:29:44 +00:00
|
|
|
screen->updateNetizenWindowDel(w->getClientWindow());
|
2002-04-11 03:20:38 +00:00
|
|
|
|
2002-05-25 00:29:44 +00:00
|
|
|
BlackboxWindowList::iterator it = windowList.begin();
|
|
|
|
const BlackboxWindowList::iterator end = windowList.end();
|
|
|
|
unsigned int i = 0;
|
|
|
|
for (; it != end; ++it, ++i)
|
2002-05-11 05:33:49 +00:00
|
|
|
(*it)->setWindowNumber(i);
|
2002-04-11 03:20:38 +00:00
|
|
|
|
2002-05-25 00:29:44 +00:00
|
|
|
if (i == 0)
|
|
|
|
cascade_x = cascade_y = 32;
|
2002-04-11 03:20:38 +00:00
|
|
|
|
2002-05-25 00:29:44 +00:00
|
|
|
return i;
|
2002-05-10 02:58:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2002-07-10 22:24:48 +00:00
|
|
|
void Workspace::focusFallback(const BlackboxWindow *old_window) {
|
|
|
|
BlackboxWindow *newfocus = 0;
|
|
|
|
|
|
|
|
// 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;
|
|
|
|
}
|
|
|
|
|
|
|
|
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;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
screen->getBlackbox()->setFocusedWindow(newfocus);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2002-04-11 03:20:38 +00:00
|
|
|
void Workspace::showAll(void) {
|
2002-05-25 00:29:44 +00:00
|
|
|
std::for_each(stackingList.begin(), stackingList.end(),
|
|
|
|
std::mem_fun(&BlackboxWindow::show));
|
2002-04-11 03:20:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Workspace::hideAll(void) {
|
2002-05-25 00:29:44 +00:00
|
|
|
// withdraw in reverse order to minimize the number of Expose events
|
2002-04-11 03:20:38 +00:00
|
|
|
|
2002-05-25 00:29:44 +00:00
|
|
|
BlackboxWindowList lst(stackingList.rbegin(), stackingList.rend());
|
2002-04-11 03:20:38 +00:00
|
|
|
|
2002-05-25 00:29:44 +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
|
|
|
}
|
|
|
|
|
|
|
|
|
2002-05-25 00:29:44 +00:00
|
|
|
void Workspace::removeAll(void) {
|
|
|
|
while (! windowList.empty())
|
|
|
|
windowList.front()->iconify();
|
|
|
|
}
|
2002-04-11 03:20:38 +00:00
|
|
|
|
|
|
|
|
2002-05-25 00:29:44 +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
|
|
|
|
2002-05-25 00:29:44 +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-05-25 00:29:44 +00:00
|
|
|
}
|
2002-04-11 03:20:38 +00:00
|
|
|
|
|
|
|
|
2002-05-25 00:29:44 +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
|
|
|
|
2002-05-25 00:29:44 +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
|
|
|
|
2002-05-25 00:29:44 +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
|
|
|
|
2002-05-25 00:29:44 +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
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2002-05-25 00:29:44 +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
|
|
|
|
2002-05-25 00:29:44 +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
|
|
|
|
2002-05-25 00:29:44 +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
|
|
|
|
2002-05-25 00:29:44 +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
|
|
|
}
|
|
|
|
|
2002-05-25 00:29:44 +00:00
|
|
|
raiseTransients(win, stack);
|
2002-04-11 03:20:38 +00:00
|
|
|
|
2002-05-25 00:29:44 +00:00
|
|
|
screen->raiseWindows(&stack_vector[0], stack_vector.size());
|
|
|
|
}
|
2002-04-11 03:20:38 +00:00
|
|
|
|
|
|
|
|
2002-05-25 00:29:44 +00:00
|
|
|
void Workspace::lowerWindow(BlackboxWindow *w) {
|
|
|
|
BlackboxWindow *win = w;
|
2002-04-11 03:20:38 +00:00
|
|
|
|
2002-05-25 00:29:44 +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();
|
|
|
|
}
|
|
|
|
|
2002-05-25 00:29:44 +00:00
|
|
|
// 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
|
|
|
|
2002-05-25 00:29:44 +00:00
|
|
|
lowerTransients(win, stack);
|
2002-04-11 03:20:38 +00:00
|
|
|
|
2002-05-25 00:29:44 +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
|
|
|
|
2002-05-25 00:29:44 +00:00
|
|
|
XLowerWindow(screen->getBaseDisplay()->getXDisplay(), stack_vector.front());
|
|
|
|
XRestackWindows(screen->getBaseDisplay()->getXDisplay(),
|
|
|
|
&stack_vector[0], stack_vector.size());
|
2002-06-21 20:40:14 +00:00
|
|
|
screen->lowerDesktops();
|
2002-04-11 03:20:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Workspace::reconfigure(void) {
|
|
|
|
clientmenu->reconfigure();
|
2002-05-25 00:29:44 +00:00
|
|
|
std::for_each(windowList.begin(), windowList.end(),
|
|
|
|
std::mem_fun(&BlackboxWindow::reconfigure));
|
|
|
|
}
|
|
|
|
|
2002-04-11 03:20:38 +00:00
|
|
|
|
2002-05-25 00:29:44 +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
|
|
|
}
|
|
|
|
|
|
|
|
|
2002-05-25 00:29:44 +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
|
|
|
}
|
|
|
|
|
|
|
|
|
2002-05-25 00:29:44 +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
|
|
|
}
|
|
|
|
|
|
|
|
|
2002-05-25 00:29:44 +00:00
|
|
|
BlackboxWindow* Workspace::getTopWindowOnStack(void) const {
|
|
|
|
return stackingList.front();
|
2002-04-11 03:20:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2002-05-25 00:29:44 +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();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2002-06-21 20:40:14 +00:00
|
|
|
void Workspace::appendStackOrder(BlackboxWindowList &stack_order) const {
|
2002-06-28 04:27:49 +00:00
|
|
|
BlackboxWindowList::const_reverse_iterator it = stackingList.rbegin();
|
|
|
|
const BlackboxWindowList::const_reverse_iterator end = stackingList.rend();
|
2002-06-21 20:40:14 +00:00
|
|
|
for (; it != end; ++it)
|
|
|
|
stack_order.push_back(*it);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2002-05-25 00:29:44 +00:00
|
|
|
bool Workspace::isCurrent(void) const {
|
|
|
|
return (id == screen->getCurrentWorkspaceID());
|
2002-04-11 03:20:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2002-05-25 00:29:44 +00:00
|
|
|
bool Workspace::isLastWindow(const BlackboxWindow* const w) const {
|
|
|
|
return (w == windowList.back());
|
|
|
|
}
|
|
|
|
|
2002-06-28 01:44:47 +00:00
|
|
|
|
2002-05-25 00:29:44 +00:00
|
|
|
void Workspace::setCurrent(void) {
|
|
|
|
screen->changeWorkspaceID(id);
|
|
|
|
}
|
|
|
|
|
2002-04-11 03:20:38 +00:00
|
|
|
|
2002-05-25 00:29:44 +00:00
|
|
|
void Workspace::setName(const string& new_name) {
|
|
|
|
if (! new_name.empty()) {
|
|
|
|
name = new_name;
|
2002-04-11 03:20:38 +00:00
|
|
|
} else {
|
2002-06-28 01:44:47 +00:00
|
|
|
// attempt to get from the _NET_WM_DESKTOP_NAMES property
|
|
|
|
XAtom::StringVect namesList;
|
|
|
|
unsigned long numnames = id + 1;
|
|
|
|
if (xatom->getValue(screen->getRootWindow(), XAtom::net_desktop_names,
|
|
|
|
XAtom::utf8, numnames, namesList) &&
|
|
|
|
namesList.size() > id) {
|
|
|
|
name = namesList[id];
|
|
|
|
} else {
|
|
|
|
string tmp =i18n(WorkspaceSet, WorkspaceDefaultNameFormat,
|
|
|
|
"Workspace %d");
|
|
|
|
assert(tmp.length() < 32);
|
|
|
|
char default_name[32];
|
|
|
|
sprintf(default_name, tmp.c_str(), id + 1);
|
|
|
|
name = default_name;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// reset the 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)) {
|
|
|
|
if (namesList.size() > id)
|
|
|
|
namesList[id] = name;
|
|
|
|
else
|
|
|
|
namesList.push_back(name);
|
2002-04-11 03:20:38 +00:00
|
|
|
}
|
2002-06-28 01:44:47 +00:00
|
|
|
xatom->setValue(screen->getRootWindow(), XAtom::net_desktop_names,
|
|
|
|
XAtom::utf8, namesList);
|
2002-05-16 22:35:59 +00:00
|
|
|
|
2002-04-11 03:20:38 +00:00
|
|
|
clientmenu->setLabel(name);
|
|
|
|
clientmenu->update();
|
2002-05-26 20:51:12 +00:00
|
|
|
screen->saveWorkspaceNames();
|
2002-04-11 03:20:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2002-05-25 00:29:44 +00:00
|
|
|
/*
|
|
|
|
* Calculate free space available for window placement.
|
|
|
|
*/
|
|
|
|
typedef std::vector<Rect> rectList;
|
2002-04-11 03:20:38 +00:00
|
|
|
|
2002-04-20 18:30:47 +00:00
|
|
|
static rectList calcSpace(const Rect &win, const rectList &spaces) {
|
2002-05-25 00:29:44 +00:00
|
|
|
Rect isect, extra;
|
2002-04-14 01:11:51 +00:00
|
|
|
rectList result;
|
2002-05-25 00:29:44 +00:00
|
|
|
rectList::const_iterator siter, end = spaces.end();
|
|
|
|
for (siter = spaces.begin(); siter != end; ++siter) {
|
|
|
|
const Rect &curr = *siter;
|
2002-04-14 01:11:51 +00:00
|
|
|
|
2002-05-25 00:29:44 +00:00
|
|
|
if(! win.intersects(curr)) {
|
|
|
|
result.push_back(curr);
|
|
|
|
continue;
|
2002-04-14 01:11:51 +00:00
|
|
|
}
|
2002-05-25 00:29:44 +00:00
|
|
|
|
|
|
|
/* 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);
|
2002-04-14 01:11:51 +00:00
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2002-05-25 00:29:44 +00:00
|
|
|
|
|
|
|
static bool rowRLBT(const Rect &first, const Rect &second) {
|
|
|
|
if (first.bottom() == second.bottom())
|
|
|
|
return first.right() > second.right();
|
|
|
|
return first.bottom() > second.bottom();
|
2002-04-21 07:58:46 +00:00
|
|
|
}
|
2002-05-16 22:35:59 +00:00
|
|
|
|
2002-05-25 00:29:44 +00:00
|
|
|
static bool rowRLTB(const Rect &first, const Rect &second) {
|
|
|
|
if (first.y() == second.y())
|
|
|
|
return first.right() > second.right();
|
|
|
|
return first.y() < second.y();
|
2002-04-21 07:58:46 +00:00
|
|
|
}
|
|
|
|
|
2002-05-25 00:29:44 +00:00
|
|
|
static bool rowLRBT(const Rect &first, const Rect &second) {
|
|
|
|
if (first.bottom() == second.bottom())
|
|
|
|
return first.x() < second.x();
|
|
|
|
return first.bottom() > second.bottom();
|
2002-04-28 01:05:12 +00:00
|
|
|
}
|
2002-05-16 22:35:59 +00:00
|
|
|
|
2002-05-25 00:29:44 +00:00
|
|
|
static bool rowLRTB(const Rect &first, const Rect &second) {
|
|
|
|
if (first.y() == second.y())
|
|
|
|
return first.x() < second.x();
|
|
|
|
return first.y() < second.y();
|
2002-04-28 01:05:12 +00:00
|
|
|
}
|
2002-05-16 22:35:59 +00:00
|
|
|
|
2002-05-25 00:29:44 +00:00
|
|
|
static bool colLRTB(const Rect &first, const Rect &second) {
|
|
|
|
if (first.x() == second.x())
|
|
|
|
return first.y() < second.y();
|
|
|
|
return first.x() < second.x();
|
2002-04-28 01:05:12 +00:00
|
|
|
}
|
2002-05-16 22:35:59 +00:00
|
|
|
|
2002-05-25 00:29:44 +00:00
|
|
|
static bool colLRBT(const Rect &first, const Rect &second) {
|
|
|
|
if (first.x() == second.x())
|
|
|
|
return first.bottom() > second.bottom();
|
|
|
|
return first.x() < second.x();
|
2002-04-21 07:58:46 +00:00
|
|
|
}
|
|
|
|
|
2002-05-25 00:29:44 +00:00
|
|
|
static bool colRLTB(const Rect &first, const Rect &second) {
|
|
|
|
if (first.right() == second.right())
|
|
|
|
return first.y() < second.y();
|
|
|
|
return first.right() > second.right();
|
2002-04-28 01:05:12 +00:00
|
|
|
}
|
2002-05-16 22:35:59 +00:00
|
|
|
|
2002-05-25 00:29:44 +00:00
|
|
|
static bool colRLBT(const Rect &first, const Rect &second) {
|
|
|
|
if (first.right() == second.right())
|
|
|
|
return first.bottom() > second.bottom();
|
|
|
|
return first.right() > second.right();
|
2002-04-21 07:58:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2002-05-25 00:29:44 +00:00
|
|
|
bool Workspace::smartPlacement(Rect& win, const Rect& availableArea) {
|
2002-04-14 01:11:51 +00:00
|
|
|
rectList spaces;
|
2002-05-25 00:29:44 +00:00
|
|
|
spaces.push_back(availableArea); //initially the entire screen is free
|
2002-05-16 22:35:59 +00:00
|
|
|
|
2002-04-14 01:11:51 +00:00
|
|
|
//Find Free Spaces
|
2002-05-30 04:35:22 +00:00
|
|
|
BlackboxWindowList::const_iterator wit = windowList.begin(),
|
|
|
|
end = windowList.end();
|
2002-05-25 00:29:44 +00:00
|
|
|
Rect tmp;
|
|
|
|
for (; wit != end; ++wit) {
|
|
|
|
const BlackboxWindow* const curr = *wit;
|
2002-05-30 04:35:22 +00:00
|
|
|
|
|
|
|
if (curr->isShaded()) continue;
|
|
|
|
|
2002-05-25 00:29:44 +00:00
|
|
|
tmp.setRect(curr->frameRect().x(), curr->frameRect().y(),
|
|
|
|
curr->frameRect().width() + screen->getBorderWidth(),
|
|
|
|
curr->frameRect().height() + screen->getBorderWidth());
|
|
|
|
|
|
|
|
spaces = calcSpace(tmp, spaces);
|
2002-04-14 01:11:51 +00:00
|
|
|
}
|
2002-04-21 07:58:46 +00:00
|
|
|
|
2002-05-25 00:29:44 +00:00
|
|
|
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);
|
2002-05-25 00:29:44 +00:00
|
|
|
else
|
2002-06-01 17:54:32 +00:00
|
|
|
std::sort(spaces.begin(), spaces.end(), rowLRBT);
|
2002-05-25 00:29:44 +00:00
|
|
|
} else {
|
|
|
|
if(screen->getColPlacementDirection() == BScreen::TopBottom)
|
2002-06-01 17:54:32 +00:00
|
|
|
std::sort(spaces.begin(), spaces.end(), rowRLTB);
|
2002-05-25 00:29:44 +00:00
|
|
|
else
|
2002-06-01 17:54:32 +00:00
|
|
|
std::sort(spaces.begin(), spaces.end(), rowRLBT);
|
2002-05-25 00:29:44 +00:00
|
|
|
}
|
|
|
|
} 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);
|
2002-05-25 00:29:44 +00:00
|
|
|
else
|
2002-06-01 17:54:32 +00:00
|
|
|
std::sort(spaces.begin(), spaces.end(), colRLTB);
|
2002-05-25 00:29:44 +00:00
|
|
|
} else {
|
|
|
|
if(screen->getRowPlacementDirection() == BScreen::LeftRight)
|
2002-06-01 17:54:32 +00:00
|
|
|
std::sort(spaces.begin(), spaces.end(), colLRBT);
|
2002-05-25 00:29:44 +00:00
|
|
|
else
|
2002-06-01 17:54:32 +00:00
|
|
|
std::sort(spaces.begin(), spaces.end(), colRLBT);
|
2002-05-25 00:29:44 +00:00
|
|
|
}
|
|
|
|
}
|
2002-05-16 22:35:59 +00:00
|
|
|
|
2002-05-25 00:29:44 +00:00
|
|
|
rectList::const_iterator sit = spaces.begin(), spaces_end = spaces.end();
|
|
|
|
for(; sit != spaces_end; ++sit) {
|
|
|
|
if (sit->width() >= win.width() && sit->height() >= win.height())
|
2002-04-28 01:05:12 +00:00
|
|
|
break;
|
2002-05-25 00:29:44 +00:00
|
|
|
}
|
2002-04-21 07:58:46 +00:00
|
|
|
|
2002-05-25 00:29:44 +00:00
|
|
|
if (sit == spaces_end)
|
|
|
|
return False;
|
2002-04-21 07:58:46 +00:00
|
|
|
|
2002-05-25 00:29:44 +00:00
|
|
|
//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;
|
|
|
|
}
|
2002-04-21 07:58:46 +00:00
|
|
|
|
2002-05-16 22:35:59 +00:00
|
|
|
|
2002-05-28 12:39:23 +00:00
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2002-05-25 00:29:44 +00:00
|
|
|
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;
|
2002-04-21 07:58:46 +00:00
|
|
|
|
2002-05-25 00:29:44 +00:00
|
|
|
if (cascade_x == 32) {
|
|
|
|
cascade_x += availableArea.x();
|
|
|
|
cascade_y += availableArea.y();
|
2002-04-27 21:15:00 +00:00
|
|
|
}
|
2002-04-16 01:10:53 +00:00
|
|
|
|
2002-05-25 00:29:44 +00:00
|
|
|
win.setPos(cascade_x, cascade_y);
|
|
|
|
|
|
|
|
return True;
|
2002-04-16 01:10:53 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2002-05-25 00:29:44 +00:00
|
|
|
void Workspace::placeWindow(BlackboxWindow *win) {
|
|
|
|
Rect availableArea(screen->availableArea()),
|
|
|
|
new_win(availableArea.x(), availableArea.y(),
|
|
|
|
win->frameRect().width(), win->frameRect().height());
|
|
|
|
bool placed = False;
|
2002-04-14 01:11:51 +00:00
|
|
|
|
2002-05-25 00:29:44 +00:00
|
|
|
switch (screen->getPlacementPolicy()) {
|
2002-04-16 01:10:53 +00:00
|
|
|
case BScreen::RowSmartPlacement:
|
|
|
|
case BScreen::ColSmartPlacement:
|
2002-05-25 00:29:44 +00:00
|
|
|
placed = smartPlacement(new_win, availableArea);
|
2002-04-28 00:11:19 +00:00
|
|
|
break;
|
2002-05-28 12:39:23 +00:00
|
|
|
case BScreen::UnderMousePlacement:
|
|
|
|
placed = underMousePlacement(new_win, availableArea);
|
2002-05-25 00:29:44 +00:00
|
|
|
default:
|
|
|
|
break; // handled below
|
2002-04-11 03:20:38 +00:00
|
|
|
} // switch
|
|
|
|
|
2002-05-25 00:29:44 +00:00
|
|
|
if (placed == False) {
|
|
|
|
cascadePlacement(new_win, availableArea);
|
|
|
|
cascade_x += win->getTitleHeight() + (screen->getBorderWidth() * 2);
|
|
|
|
cascade_y += win->getTitleHeight() + (screen->getBorderWidth() * 2);
|
|
|
|
}
|
2002-04-16 01:10:53 +00:00
|
|
|
|
2002-05-25 00:29:44 +00:00
|
|
|
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());
|
2002-04-15 02:37:18 +00:00
|
|
|
}
|