fluxbox/src/FbTk/EventManager.cc
2005-01-24 18:02:34 +00:00

203 lines
5.4 KiB
C++

// EventManager.cc
// Copyright (c) 2002 - 2005 Henrik Kinnunen (fluxgen at fluxbox dot org)
//
// Permission is hereby granted, free of charge, to any person obtaining a
// copy of this software and associated documentation files (the "Software"),
// to deal in the Software without restriction, including without limitation
// the rights to use, copy, modify, merge, publish, distribute, sublicense,
// and/or sell copies of the Software, and to permit persons to whom the
// Software is furnished to do so, subject to the following conditions:
//
// The above copyright notice and this permission notice shall be included in
// all copies or substantial portions of the Software.
//
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
// THE AUTHORS 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 "EventManager.hh"
#include "FbWindow.hh"
#include "App.hh"
#include "I18n.hh"
#include <iostream>
using namespace std;
namespace FbTk {
EventManager *EventManager::instance() {
static EventManager ev;
return &ev;
}
EventManager::~EventManager() {
#ifdef DEBUG
if (m_eventhandlers.size() != 0)
cerr<<"FbTk::EventManager: Warning: unregistered eventhandlers!"<<endl;
#endif // DEBUG
}
void EventManager::handleEvent(XEvent &ev) {
dispatch(ev.xany.window, ev);
}
void EventManager::add(EventHandler &ev, const FbWindow &win) {
registerEventHandler(ev, win.window());
}
void EventManager::addParent(EventHandler &ev, const FbWindow &win) {
if (win.window() != 0)
m_parent[win.window()] = &ev;
}
void EventManager::remove(const FbWindow &win) {
unregisterEventHandler(win.window());
}
EventHandler *EventManager::find(Window win) {
return m_eventhandlers[win];
}
Window EventManager::getEventWindow(XEvent &ev) {
// we only have cases for events that differ from xany
switch (ev.type) {
case CreateNotify:
// XCreateWindowEvent
return ev.xcreatewindow.window;
break;
case DestroyNotify:
// XDestroyWindowEvent
return ev.xdestroywindow.window;
break;
case UnmapNotify:
// XUnmapEvent
return ev.xunmap.window;
break;
case MapNotify:
// XMapEvent
return ev.xmap.window;
break;
case MapRequest:
// XMapRequestEvent
return ev.xmaprequest.window;
break;
case ReparentNotify:
// XReparentEvent
return ev.xreparent.window;
break;
case ConfigureNotify:
// XConfigureNotify
return ev.xconfigure.window;
break;
case GravityNotify:
// XGravityNotify
return ev.xgravity.window;
break;
case ConfigureRequest:
// XConfigureRequestEvent
return ev.xconfigurerequest.window;
break;
case CirculateNotify:
// XCirculateEvent
return ev.xcirculate.window;
break;
case CirculateRequest:
// XCirculateRequestEvent
return ev.xcirculaterequest.window;
break;
default:
return ev.xany.window;
}
}
void EventManager::registerEventHandler(EventHandler &ev, Window win) {
if (win != None)
m_eventhandlers[win] = &ev;
}
void EventManager::unregisterEventHandler(Window win) {
if (win != None) {
m_eventhandlers.erase(win);
m_parent.erase(win);
}
}
void EventManager::dispatch(Window win, XEvent &ev, bool parent) {
EventHandler *evhand = 0;
if (parent) {
EventHandlerMap::iterator it = m_parent.find(win);
if (it == m_parent.end())
return;
else
evhand = it->second;
} else {
win = getEventWindow(ev);
EventHandlerMap::iterator it = m_eventhandlers.find(win);
if (it == m_eventhandlers.end())
return;
else
evhand = it->second;
}
if (evhand == 0)
return;
switch (ev.type) {
case KeyPress:
evhand->keyPressEvent(ev.xkey);
break;
case KeyRelease:
evhand->keyReleaseEvent(ev.xkey);
break;
case ButtonPress:
evhand->buttonPressEvent(ev.xbutton);
break;
case ButtonRelease:
evhand->buttonReleaseEvent(ev.xbutton);
break;
case MotionNotify:
evhand->motionNotifyEvent(ev.xmotion);
break;
case Expose:
evhand->exposeEvent(ev.xexpose);
break;
case EnterNotify:
evhand->enterNotifyEvent(ev.xcrossing);
break;
case LeaveNotify:
evhand->leaveNotifyEvent(ev.xcrossing);
break;
default:
evhand->handleEvent(ev);
break;
};
// find out which window is the parent and
// dispatch event
Window root, parent_win, *children = 0;
unsigned int num_children;
if (XQueryTree(FbTk::App::instance()->display(), win,
&root, &parent_win, &children, &num_children) != 0) {
if (children != 0)
XFree(children);
if (parent_win != 0 &&
parent_win != root) {
if (m_parent[parent_win] == 0)
return;
// dispatch event to parent
dispatch(parent_win, ev, true);
}
}
}
}; // end namespace FbTk