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"
|
2002-07-22 06:02:29 +00:00
|
|
|
#include "Font.hh"
|
2002-05-25 00:29:44 +00:00
|
|
|
#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-07-22 06:02:29 +00:00
|
|
|
cascade_x = cascade_y = 0;
|
|
|
|
#ifdef XINERAMA
|
|
|
|
cascade_region = 0;
|
|
|
|
#endif // XINERAMA
|
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-07-14 20:54:03 +00:00
|
|
|
readName();
|
2002-04-11 03:20:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2002-07-27 18:17:45 +00:00
|
|
|
void Workspace::addWindow(BlackboxWindow *w, bool place, bool sticky) {
|
2002-05-25 00:29:44 +00:00
|
|
|
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-05-25 00:29:44 +00:00
|
|
|
stackingList.push_front(w);
|
2002-08-09 00:44:35 +00:00
|
|
|
|
2002-07-16 08:00:59 +00:00
|
|
|
if (w->isNormal()) {
|
2002-07-27 18:17:45 +00:00
|
|
|
if (! sticky) {
|
|
|
|
w->setWorkspace(id);
|
|
|
|
w->setWindowNumber(windowList.size());
|
|
|
|
}
|
2002-04-11 03:20:38 +00:00
|
|
|
|
2002-07-16 08:00:59 +00:00
|
|
|
windowList.push_back(w);
|
2002-04-11 03:20:38 +00:00
|
|
|
|
2002-07-16 08:00:59 +00:00
|
|
|
clientmenu->insert(w->getTitle());
|
|
|
|
clientmenu->update();
|
|
|
|
|
2002-07-27 18:17:45 +00:00
|
|
|
if (! sticky)
|
|
|
|
screen->updateNetizenWindowAdd(w->getClientWindow(), id);
|
|
|
|
|
|
|
|
if (screen->doFocusNew() || (w->isTransient() && w->getTransientFor() &&
|
|
|
|
w->getTransientFor()->isFocused())) {
|
2002-08-13 21:14:37 +00:00
|
|
|
if (id != screen->getCurrentWorkspaceID()) {
|
2002-07-27 18:17:45 +00:00
|
|
|
/*
|
|
|
|
not on the focused workspace, so the window is not going to get focus
|
|
|
|
but if the user wants new windows focused, then it should get focus
|
|
|
|
when this workspace does become focused.
|
|
|
|
*/
|
|
|
|
lastfocus = w;
|
|
|
|
}
|
2002-07-20 20:44:29 +00:00
|
|
|
}
|
2002-07-16 08:00:59 +00:00
|
|
|
}
|
2002-04-11 03:20:38 +00:00
|
|
|
|
2002-07-16 08:00:59 +00:00
|
|
|
if (! w->isDesktop())
|
|
|
|
raiseWindow(w);
|
|
|
|
else
|
|
|
|
lowerWindow(w);
|
2002-04-11 03:20:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2002-07-27 18:17:45 +00:00
|
|
|
void Workspace::removeWindow(BlackboxWindow *w, bool sticky) {
|
2002-05-25 00:29:44 +00:00
|
|
|
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()) {
|
2002-07-14 18:45:46 +00:00
|
|
|
focusFallback(w);
|
2002-07-27 18:17:45 +00:00
|
|
|
}
|
|
|
|
|
2002-07-16 08:00:59 +00:00
|
|
|
if (! w->isNormal()) return;
|
|
|
|
|
2002-07-27 18:17:45 +00:00
|
|
|
BlackboxWindowList::iterator it, end = windowList.end();
|
|
|
|
int i;
|
|
|
|
for (i = 0, it = windowList.begin(); it != end; ++it, ++i)
|
|
|
|
if (*it == w)
|
|
|
|
break;
|
|
|
|
assert(it != end);
|
|
|
|
|
|
|
|
windowList.erase(it);
|
|
|
|
clientmenu->remove(i);
|
2002-04-11 03:20:38 +00:00
|
|
|
clientmenu->update();
|
|
|
|
|
2002-07-27 18:17:45 +00:00
|
|
|
if (! sticky) {
|
|
|
|
screen->updateNetizenWindowDel(w->getClientWindow());
|
2002-04-11 03:20:38 +00:00
|
|
|
|
2002-07-27 18:17:45 +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
|
|
|
|
2002-07-22 06:02:29 +00:00
|
|
|
if (i == 0) {
|
|
|
|
cascade_x = cascade_y = 0;
|
|
|
|
#ifdef XINERAMA
|
|
|
|
cascade_region = 0;
|
|
|
|
#endif // XINERAMA
|
|
|
|
}
|
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;
|
|
|
|
|
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;
|
2002-07-16 08:00:59 +00:00
|
|
|
if (tmp && tmp->isNormal() && tmp->setInputFocus()) {
|
2002-07-14 18:45:46 +00:00
|
|
|
// 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-07-27 18:17:45 +00:00
|
|
|
void Workspace::setFocused(const BlackboxWindow *w, bool focused) {
|
|
|
|
BlackboxWindowList::iterator it, end = windowList.end();
|
|
|
|
int i;
|
|
|
|
for (i = 0, it = windowList.begin(); it != end; ++it, ++i)
|
|
|
|
if (*it == w)
|
|
|
|
break;
|
2002-07-28 17:37:31 +00:00
|
|
|
// if its == end, then a window thats not in the windowList
|
|
|
|
// got focused, such as a !isNormal() window.
|
|
|
|
if (it != end)
|
|
|
|
clientmenu->setItemSelected(i, focused);
|
2002-07-27 18:17:45 +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
|
|
|
|
*/
|
2002-08-07 00:24:58 +00:00
|
|
|
static unsigned int countTransients(const BlackboxWindow * const win) {
|
|
|
|
BlackboxWindowList transients = win->getTransients();
|
|
|
|
if (transients.empty()) return 0;
|
|
|
|
|
|
|
|
unsigned int ret = transients.size();
|
|
|
|
BlackboxWindowList::const_iterator it = transients.begin(),
|
|
|
|
end = transients.end();
|
|
|
|
for (; it != end; ++it)
|
|
|
|
ret += countTransients(*it);
|
|
|
|
|
2002-05-25 00:29:44 +00:00
|
|
|
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) {
|
2002-08-07 00:24:58 +00:00
|
|
|
if (win->getTransients().empty()) return; // nothing to do
|
2002-05-25 00:29:44 +00:00
|
|
|
|
|
|
|
// put win's transients in the stack
|
|
|
|
BlackboxWindowList::const_iterator it, end = win->getTransients().end();
|
|
|
|
for (it = win->getTransients().begin(); it != end; ++it) {
|
2002-08-07 00:24:58 +00:00
|
|
|
BlackboxWindow *w = *it;
|
|
|
|
*stack++ = w->getFrameWindow();
|
|
|
|
screen->updateNetizenWindowRaise(w->getClientWindow());
|
|
|
|
|
|
|
|
if (! w->isIconic()) {
|
|
|
|
Workspace *wkspc = screen->getWorkspace(w->getWorkspaceNumber());
|
|
|
|
wkspc->stackingList.remove(w);
|
|
|
|
wkspc->stackingList.push_front(w);
|
2002-05-25 00:29:44 +00:00
|
|
|
}
|
|
|
|
}
|
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
|
2002-08-07 00:24:58 +00:00
|
|
|
for (it = win->getTransients().begin(); it != end; ++it)
|
2002-05-25 00:29:44 +00:00
|
|
|
raiseTransients(*it, stack);
|
|
|
|
}
|
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) {
|
2002-08-07 00:24:58 +00:00
|
|
|
if (win->getTransients().empty()) 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();
|
2002-08-07 00:24:58 +00:00
|
|
|
for (it = win->getTransients().rbegin(); it != end; ++it)
|
2002-05-25 00:29:44 +00:00
|
|
|
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) {
|
2002-08-07 00:24:58 +00:00
|
|
|
BlackboxWindow *w = *it;
|
|
|
|
*stack++ = w->getFrameWindow();
|
|
|
|
screen->updateNetizenWindowLower(w->getClientWindow());
|
|
|
|
|
|
|
|
if (! w->isIconic()) {
|
|
|
|
Workspace *wkspc = screen->getWorkspace(w->getWorkspaceNumber());
|
|
|
|
wkspc->stackingList.remove(w);
|
|
|
|
wkspc->stackingList.push_back(w);
|
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::raiseWindow(BlackboxWindow *w) {
|
|
|
|
BlackboxWindow *win = w;
|
|
|
|
|
2002-07-16 08:00:59 +00:00
|
|
|
if (win->isDesktop()) return;
|
|
|
|
|
2002-05-25 00:29:44 +00:00
|
|
|
// walk up the transient_for's to the window that is not a transient
|
2002-08-07 00:24:58 +00:00
|
|
|
while (win->isTransient() && win->getTransientFor())
|
2002-05-25 00:29:44 +00:00
|
|
|
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());
|
2002-07-16 08:00:59 +00:00
|
|
|
if (! (win->isIconic() || win->isDesktop())) {
|
2002-05-25 00:29:44 +00:00
|
|
|
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
|
2002-08-07 00:24:58 +00:00
|
|
|
while (win->isTransient() && win->getTransientFor())
|
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());
|
2002-07-16 08:00:59 +00:00
|
|
|
if (! (win->isIconic() || win->isDesktop())) {
|
2002-05-25 00:29:44 +00:00
|
|
|
Workspace *wkspc = screen->getWorkspace(win->getWorkspaceNumber());
|
|
|
|
wkspc->stackingList.remove(win);
|
|
|
|
wkspc->stackingList.push_back(win);
|
|
|
|
}
|
2002-04-11 03:20:38 +00:00
|
|
|
|
2002-07-13 21:25:19 +00:00
|
|
|
screen->lowerWindows(&stack_vector[0], stack_vector.size());
|
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();
|
2002-08-07 00:24:58 +00:00
|
|
|
while (index-- > 0) // increment to index
|
|
|
|
++it;
|
2002-05-25 00:29:44 +00:00
|
|
|
return *it;
|
|
|
|
}
|
2002-08-07 00:24:58 +00:00
|
|
|
|
2002-05-25 00:29:44 +00:00
|
|
|
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 {
|
2002-08-07 00:24:58 +00:00
|
|
|
assert(! stackingList.empty());
|
2002-05-25 00:29:44 +00:00
|
|
|
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)
|
2002-07-16 22:01:17 +00:00
|
|
|
if ((*it)->isNormal())
|
2002-07-16 22:00:44 +00:00
|
|
|
stack_order.push_back(*it);
|
2002-06-21 20:40:14 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2002-08-07 00:24:58 +00:00
|
|
|
void Workspace::hide(void) {
|
|
|
|
BlackboxWindow *focused = screen->getBlackbox()->getFocusedWindow();
|
|
|
|
if (focused && focused->getScreen() == screen) {
|
|
|
|
assert(focused->isStuck() || focused->getWorkspaceNumber() == id);
|
|
|
|
|
|
|
|
lastfocus = focused;
|
|
|
|
} else {
|
|
|
|
// if no window had focus, no need to store a last focus
|
|
|
|
lastfocus = (BlackboxWindow *) 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
// when we switch workspaces, unfocus whatever was focused
|
|
|
|
screen->getBlackbox()->setFocusedWindow((BlackboxWindow *) 0);
|
|
|
|
|
|
|
|
// withdraw windows in reverse order to minimize the number of Expose events
|
|
|
|
|
|
|
|
BlackboxWindowList::reverse_iterator it = stackingList.rbegin();
|
|
|
|
const BlackboxWindowList::reverse_iterator end = stackingList.rend();
|
|
|
|
for (; it != end; ++it) {
|
|
|
|
BlackboxWindow *bw = *it;
|
|
|
|
// not normal windows cant focus from mouse enters anyways, so we dont
|
|
|
|
// need to unmap/remap them on workspace changes
|
|
|
|
if (! bw->isStuck() || bw->isNormal())
|
|
|
|
bw->withdraw();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Workspace::show(void) {
|
2002-08-08 05:00:14 +00:00
|
|
|
BlackboxWindowList::iterator it = stackingList.begin();
|
|
|
|
const BlackboxWindowList::iterator end = stackingList.end();
|
|
|
|
for (; it != end; ++it) {
|
|
|
|
BlackboxWindow *bw = *it;
|
|
|
|
// not normal windows cant focus from mouse enters anyways, so we dont
|
|
|
|
// need to unmap/remap them on workspace changes
|
|
|
|
if (! bw->isStuck() || bw->isNormal())
|
|
|
|
bw->show();
|
|
|
|
}
|
2002-08-07 00:24:58 +00:00
|
|
|
|
|
|
|
XSync(screen->getBlackbox()->getXDisplay(), False);
|
|
|
|
|
|
|
|
if (screen->doFocusLast()) {
|
|
|
|
if (! screen->isSloppyFocus() && ! lastfocus && ! stackingList.empty())
|
|
|
|
lastfocus = stackingList.front();
|
|
|
|
|
|
|
|
if (lastfocus)
|
|
|
|
lastfocus->setInputFocus();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
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-07-14 20:54:03 +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 {
|
2002-07-14 20:54:03 +00:00
|
|
|
/*
|
|
|
|
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
|
2002-06-28 01:44:47 +00:00
|
|
|
}
|
2002-07-14 20:54:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Workspace::setName(const string& new_name) {
|
|
|
|
// set the _NET_WM_DESKTOP_NAMES property with the new name
|
2002-06-28 01:44:47 +00:00
|
|
|
XAtom::StringVect namesList;
|
|
|
|
unsigned long numnames = (unsigned) -1;
|
|
|
|
if (xatom->getValue(screen->getRootWindow(), XAtom::net_desktop_names,
|
2002-07-14 17:30:26 +00:00
|
|
|
XAtom::utf8, numnames, namesList) &&
|
|
|
|
namesList.size() > id)
|
2002-07-14 20:54:03 +00:00
|
|
|
namesList[id] = new_name;
|
2002-07-14 17:30:26 +00:00
|
|
|
else
|
2002-07-14 20:54:03 +00:00
|
|
|
namesList.push_back(new_name);
|
2002-07-14 17:30:26 +00:00
|
|
|
|
2002-06-28 01:44:47 +00:00
|
|
|
xatom->setValue(screen->getRootWindow(), XAtom::net_desktop_names,
|
|
|
|
XAtom::utf8, namesList);
|
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-07-22 04:29:40 +00:00
|
|
|
bool Workspace::smartPlacement(Rect& win) {
|
2002-04-14 01:11:51 +00:00
|
|
|
rectList spaces;
|
2002-07-22 04:29:40 +00:00
|
|
|
|
|
|
|
//initially the entire screen is free
|
|
|
|
#ifdef XINERAMA
|
|
|
|
if (screen->isXineramaActive() &&
|
|
|
|
screen->getBlackbox()->doXineramaPlacement()) {
|
|
|
|
RectList availableAreas = screen->allAvailableAreas();
|
|
|
|
RectList::iterator it, end = availableAreas.end();
|
|
|
|
|
|
|
|
for (it = availableAreas.begin(); it != end; ++it)
|
|
|
|
spaces.push_back(*it);
|
|
|
|
} else
|
|
|
|
#endif // XINERAMA
|
|
|
|
spaces.push_back(screen->availableArea());
|
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
|
|
|
|
2002-07-23 20:54:40 +00:00
|
|
|
// watch for shaded windows and full-maxed windows
|
|
|
|
if (curr->isShaded()) {
|
|
|
|
if (screen->getPlaceIgnoreShaded()) continue;
|
|
|
|
} else if (curr->isMaximizedFull()) {
|
|
|
|
if (screen->getPlaceIgnoreMaximized()) continue;
|
|
|
|
}
|
2002-05-30 04:35:22 +00:00
|
|
|
|
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-07-22 06:11:33 +00:00
|
|
|
bool Workspace::underMousePlacement(Rect &win) {
|
2002-05-28 12:39:23 +00:00
|
|
|
int x, y, rx, ry;
|
|
|
|
Window c, r;
|
|
|
|
unsigned int m;
|
|
|
|
XQueryPointer(screen->getBlackbox()->getXDisplay(), screen->getRootWindow(),
|
|
|
|
&r, &c, &rx, &ry, &x, &y, &m);
|
2002-07-22 04:29:40 +00:00
|
|
|
|
|
|
|
Rect area;
|
|
|
|
#ifdef XINERAMA
|
|
|
|
if (screen->isXineramaActive() &&
|
|
|
|
screen->getBlackbox()->doXineramaPlacement()) {
|
|
|
|
RectList availableAreas = screen->allAvailableAreas();
|
|
|
|
RectList::iterator it, end = availableAreas.end();
|
|
|
|
|
|
|
|
for (it = availableAreas.begin(); it != end; ++it)
|
|
|
|
if (it->contains(rx, ry)) break;
|
|
|
|
assert(it != end); // the mouse isn't inside an area?
|
|
|
|
area = *it;
|
|
|
|
} else
|
|
|
|
#endif // XINERAMA
|
|
|
|
area = screen->availableArea();
|
|
|
|
|
2002-05-28 12:39:23 +00:00
|
|
|
x = rx - win.width() / 2;
|
|
|
|
y = ry - win.height() / 2;
|
|
|
|
|
2002-07-22 04:29:40 +00:00
|
|
|
if (x < area.x())
|
|
|
|
x = area.x();
|
|
|
|
if (y < area.y())
|
|
|
|
y = area.y();
|
|
|
|
if (x + win.width() > area.x() + area.width())
|
|
|
|
x = area.x() + area.width() - win.width();
|
|
|
|
if (y + win.height() > area.y() + area.height())
|
|
|
|
y = area.y() + area.height() - win.height();
|
2002-05-28 12:39:23 +00:00
|
|
|
|
|
|
|
win.setX(x);
|
|
|
|
win.setY(y);
|
|
|
|
|
|
|
|
return True;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2002-07-22 06:02:29 +00:00
|
|
|
bool Workspace::cascadePlacement(Rect &win, const int offset) {
|
|
|
|
Rect area;
|
|
|
|
|
|
|
|
#ifdef XINERAMA
|
|
|
|
if (screen->isXineramaActive() &&
|
|
|
|
screen->getBlackbox()->doXineramaPlacement()) {
|
|
|
|
area = screen->allAvailableAreas()[cascade_region];
|
|
|
|
} else
|
|
|
|
#endif // XINERAMA
|
|
|
|
area = screen->availableArea();
|
2002-07-22 04:29:40 +00:00
|
|
|
|
2002-07-22 06:02:29 +00:00
|
|
|
if ((static_cast<signed>(cascade_x + win.width()) > area.right() + 1) ||
|
|
|
|
(static_cast<signed>(cascade_y + win.height()) > area.bottom() + 1)) {
|
|
|
|
cascade_x = cascade_y = 0;
|
|
|
|
#ifdef XINERAMA
|
|
|
|
if (screen->isXineramaActive() &&
|
|
|
|
screen->getBlackbox()->doXineramaPlacement()) {
|
|
|
|
// go to the next xinerama region, and use its area
|
|
|
|
if (++cascade_region >= screen->allAvailableAreas().size())
|
|
|
|
cascade_region = 0;
|
|
|
|
area = screen->allAvailableAreas()[cascade_region];
|
|
|
|
}
|
|
|
|
#endif // XINERAMA
|
|
|
|
}
|
2002-04-21 07:58:46 +00:00
|
|
|
|
2002-07-22 06:02:29 +00:00
|
|
|
if (cascade_x == 0) {
|
|
|
|
cascade_x = area.x() + offset;
|
|
|
|
cascade_y = area.y() + offset;
|
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);
|
|
|
|
|
2002-07-22 06:02:29 +00:00
|
|
|
cascade_x += offset;
|
|
|
|
cascade_y += offset;
|
|
|
|
|
2002-05-25 00:29:44 +00:00
|
|
|
return True;
|
2002-04-16 01:10:53 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2002-05-25 00:29:44 +00:00
|
|
|
void Workspace::placeWindow(BlackboxWindow *win) {
|
2002-07-22 04:29:40 +00:00
|
|
|
Rect new_win(0, 0, win->frameRect().width(), win->frameRect().height());
|
2002-05-25 00:29:44 +00:00
|
|
|
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-07-22 04:29:40 +00:00
|
|
|
placed = smartPlacement(new_win);
|
2002-04-28 00:11:19 +00:00
|
|
|
break;
|
2002-05-28 12:39:23 +00:00
|
|
|
case BScreen::UnderMousePlacement:
|
2002-07-16 05:00:09 +00:00
|
|
|
case BScreen::ClickMousePlacement:
|
2002-07-22 04:29:40 +00:00
|
|
|
placed = underMousePlacement(new_win);
|
2002-05-25 00:29:44 +00:00
|
|
|
default:
|
|
|
|
break; // handled below
|
2002-04-11 03:20:38 +00:00
|
|
|
} // switch
|
|
|
|
|
2002-07-22 06:02:29 +00:00
|
|
|
if (placed == False)
|
|
|
|
cascadePlacement(new_win, (win->getTitleHeight() +
|
|
|
|
screen->getBorderWidth() * 2));
|
2002-04-16 01:10:53 +00:00
|
|
|
|
2002-07-22 23:32:30 +00:00
|
|
|
if (new_win.right() > screen->availableArea().right())
|
|
|
|
new_win.setX(screen->availableArea().left());
|
|
|
|
if (new_win.bottom() > screen->availableArea().bottom())
|
|
|
|
new_win.setY(screen->availableArea().top());
|
2002-07-22 04:29:40 +00:00
|
|
|
|
2002-05-25 00:29:44 +00:00
|
|
|
win->configure(new_win.x(), new_win.y(), new_win.width(), new_win.height());
|
2002-04-15 02:37:18 +00:00
|
|
|
}
|