fluxbox/src/Container.cc
simonb c2ec3065f9 change fbwinframe to use container
move a few frame ops to have most code in Container
2005-06-19 12:12:54 +00:00

407 lines
11 KiB
C++

// Container.cc
// Copyright (c) 2003 - 2005 Henrik Kinnunen (fluxgen at fluxbox dot org)
// and Simon Bowden (rathnor at users.sourceforge.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.
// $Id$
#include "Container.hh"
#include "FbTk/Button.hh"
#include "FbTk/EventManager.hh"
#include "CompareWindow.hh"
Container::Container(const FbTk::FbWindow &parent):
FbTk::FbWindow(parent, 0, 0, 1, 1, ExposureMask),
m_align(RELATIVE),
m_max_size_per_client(60),
m_selected(0),
m_update_lock(false) {
FbTk::EventManager::instance()->add(*this, *this);
}
Container::~Container() {
// ~FbWindow cleans event manager
}
void Container::resize(unsigned int width, unsigned int height) {
// do we need to resize?
if (FbTk::FbWindow::width() == width &&
FbTk::FbWindow::height() == height)
return;
FbTk::FbWindow::resize(width, height);
repositionItems();
}
void Container::moveResize(int x, int y,
unsigned int width, unsigned int height) {
FbTk::FbWindow::moveResize(x, y, width, height);
repositionItems();
}
void Container::insertItems(ItemList &item_list, int pos) {
// make sure all items have parent == this
ItemList::iterator it = m_item_list.begin();
ItemList::iterator it_end = m_item_list.end();
for (; it != it_end; ++it) {
if ((*it)->parent() != this)
return;
}
if (pos > size() || pos < 0) {
// insert last
m_item_list.splice(m_item_list.end(), item_list);
} else if (pos == 0) {
// insert first
m_item_list.splice(m_item_list.begin(), item_list);
} else {
// find insert point
for (it = m_item_list.begin(); pos != 0; ++it, --pos)
continue;
m_item_list.splice(it, item_list);
}
m_item_list.unique();
// update position
repositionItems();
}
void Container::insertItem(Item item, int pos) {
if (find(item) != -1)
return;
// it must be a child of this window
if (item->parent() != this)
return;
if (pos >= size() || pos < 0) {
m_item_list.push_back(item);
} else if (pos == 0) {
m_item_list.push_front(item);
} else {
ItemList::iterator it = m_item_list.begin();
for (; pos != 0; ++it, --pos)
continue;
m_item_list.insert(it, item);
}
// make sure we dont have duplicate items
m_item_list.unique();
repositionItems();
}
void Container::moveItem(Item item, int movement) {
int index = find(item);
if (index < 0)
return;
int size = m_item_list.size();
int newindex = (index + movement) % size;
if (newindex < 0) // neg wrap
newindex += size;
if (newindex > index) // one smaller now
--newindex;
ItemList::iterator it = m_item_list.begin();
for (; newindex > 0 && index > 0; ++it, --newindex, --index) {
if (newindex == 0) {
m_item_list.insert(it, item);
++newindex;
}
if (index == 0) {
m_item_list.erase(it);
--index;
}
}
m_item_list.insert(it, item);
insertItem(item, newindex);
repositionItems();
}
// returns true if something was done
bool Container::moveItemTo(Item item, int x, int y) {
Window parent_return=0,
root_return=0,
*children_return = NULL;
unsigned int nchildren_return;
// get the root window
if (!XQueryTree(display(), window(),
&root_return, &parent_return, &children_return, &nchildren_return))
parent_return = parent_return;
if (children_return != NULL)
XFree(children_return);
int dest_x = 0, dest_y = 0;
Window itemwin = 0;
if (!XTranslateCoordinates(display(),
root_return, window(),
x, y, &dest_x, &dest_y,
&itemwin))
return false;
ItemList::iterator it = find_if(m_item_list.begin(),
m_item_list.end(),
CompareWindow(&FbTk::FbWindow::window,
itemwin));
// not found :(
if (it == m_item_list.end())
return false;
Window child_return = 0;
//make x and y relative to our item
if (!XTranslateCoordinates(display(),
window(), itemwin,
dest_x, dest_y, &x, &y,
&child_return))
return false;
return true;
}
bool Container::removeItem(Item item) {
ItemList::iterator it = m_item_list.begin();
ItemList::iterator it_end = m_item_list.end();
for (; it != it_end && (*it) != item; ++it);
if (it == it_end)
return false;
m_item_list.erase(it);
repositionItems();
return true;
}
bool Container::removeItem(int index) {
if (index < 0 || index > size())
return false;
ItemList::iterator it = m_item_list.begin();
for (; index != 0; ++it, --index)
continue;
if (*it == selected())
m_selected = 0;
m_item_list.erase(it);
repositionItems();
return true;
}
void Container::removeAll() {
m_selected = 0;
m_item_list.clear();
if (!m_update_lock) {
clear();
}
}
int Container::find(ConstItem item) {
ItemList::iterator it = m_item_list.begin();
ItemList::iterator it_end = m_item_list.end();
int index = 0;
for (; it != it_end; ++it, ++index) {
if ((*it) == item)
break;
}
if (it == it_end)
return -1;
return index;
}
void Container::setSelected(int pos) {
if (pos < 0 || pos >= size())
m_selected = 0;
else {
ItemList::iterator it = m_item_list.begin();
for (; pos != 0; --pos, ++it)
continue;
m_selected = *it;
// caller does any graphics stuff if appropriate
}
}
void Container::setMaxSizePerClient(unsigned int size) {
m_max_size_per_client = size;
}
void Container::setAlignment(Container::Alignment a) {
m_align = a;
}
void Container::exposeEvent(XExposeEvent &event) {
if (!m_update_lock) {
clearArea(event.x, event.y, event.width, event.height);
}
}
bool Container::tryExposeEvent(XExposeEvent &event) {
if (event.window == window()) {
exposeEvent(event);
return true;
}
ItemList::iterator it = find_if(m_item_list.begin(),
m_item_list.end(),
CompareWindow(&FbTk::FbWindow::window,
event.window));
// not found :(
if (it == m_item_list.end())
return false;
(*it)->exposeEvent(event);
return true;
}
bool Container::tryButtonPressEvent(XButtonEvent &event) {
if (event.window == window()) {
// we don't have a buttonrelease event atm
return true;
}
ItemList::iterator it = find_if(m_item_list.begin(),
m_item_list.end(),
CompareWindow(&FbTk::FbWindow::window,
event.window));
// not found :(
if (it == m_item_list.end())
return false;
(*it)->buttonPressEvent(event);
return true;
}
bool Container::tryButtonReleaseEvent(XButtonEvent &event) {
if (event.window == window()) {
// we don't have a buttonrelease event atm
return true;
}
ItemList::iterator it = find_if(m_item_list.begin(),
m_item_list.end(),
CompareWindow(&FbTk::FbWindow::window,
event.window));
// not found :(
if (it == m_item_list.end())
return false;
(*it)->buttonReleaseEvent(event);
return true;
}
void Container::repositionItems() {
if (empty() || m_update_lock)
return;
//!! TODO vertical position
const int max_width_per_client = maxWidthPerClient();
ItemList::iterator it = m_item_list.begin();
const ItemList::iterator it_end = m_item_list.end();
int borderW = m_item_list.front()->borderWidth();
int rounding_error = width() - ((maxWidthPerClient() + borderW)* m_item_list.size() - borderW);
int next_x = -borderW; // zero so the border of the first shows
int extra = 0;
int direction = 1;
if (alignment() == RIGHT) {
direction = -1;
next_x = width() - max_width_per_client + borderW;
}
for (; it != it_end; ++it, next_x += direction*(max_width_per_client + borderW + extra)) {
// we only need to do error stuff with alignment RELATIVE
if (rounding_error != 0 && alignment() == RELATIVE) {
--rounding_error;
extra = 1;
} else {
extra = 0;
}
// resize each clients including border in size
(*it)->moveResize(next_x,
-borderW,
max_width_per_client + extra,
height());
// moveresize does a clear
}
}
unsigned int Container::maxWidthPerClient() const {
switch (alignment()) {
case RIGHT:
case LEFT:
return m_max_size_per_client;
break;
case RELATIVE:
int count = size();
if (count == 0)
return width();
else {
int borderW = m_item_list.front()->borderWidth();
// there're count-1 borders to fit in with the windows
// -> 1 per window plus end
return (width() - (count - 1) * borderW) / count;
}
break;
}
// this will never happen anyway
return 1;
}
void Container::for_each(std::mem_fun_t<void, FbTk::FbWindow> function) {
std::for_each(m_item_list.begin(),
m_item_list.end(),
function);
}
void Container::setAlpha(unsigned char alpha) {
ItemList::iterator it = m_item_list.begin();
ItemList::iterator it_end = m_item_list.end();
for (; it != it_end; ++it)
(*it)->setAlpha(alpha);
}
void Container::clear() {
ItemList::iterator it = m_item_list.begin();
ItemList::iterator it_end = m_item_list.end();
for (; it != it_end; ++it)
(*it)->clear();
}