X-Git-Url: https://git.dogcows.com/gitweb?a=blobdiff_plain;f=src%2Fbindings.cc;h=2b81381c080925b48d15457531b150df9720a2d8;hb=86a2bed6595cdc926dccb4a7c0f984fd5996e3c2;hp=3e9358d5bd33edcc4829f2f6f6c161066e7fd90d;hpb=7db3ffecc980821ada3e805e2471716896e2410a;p=chaz%2Fopenbox diff --git a/src/bindings.cc b/src/bindings.cc index 3e9358d5..2b81381c 100644 --- a/src/bindings.cc +++ b/src/bindings.cc @@ -142,7 +142,7 @@ OBBindings::OBBindings() : _curpos(&_keytree), _resetkey(0,0), _timer(Openbox::instance->timerManager(), - (otk::OBTimeoutHandler)reset, this) + (otk::OBTimeoutHandler)resetChains, this) { _timer.setTimeout(5000); // chains reset after 5 seconds @@ -153,7 +153,8 @@ OBBindings::OBBindings() OBBindings::~OBBindings() { grabKeys(false); - removeAll(); + removeAllKeys(); + removeAllButtons(); } @@ -215,7 +216,7 @@ PyObject *OBBindings::find(KeyBindingTree *search, bool *conflict) const { } -bool OBBindings::add(const StringVect &keylist, PyObject *callback) +bool OBBindings::addKey(const StringVect &keylist, PyObject *callback) { KeyBindingTree *tree; bool conflict; @@ -242,7 +243,7 @@ bool OBBindings::add(const StringVect &keylist, PyObject *callback) } -bool OBBindings::remove(const StringVect &keylist) +bool OBBindings::removeKey(const StringVect &keylist) { assert(false); // XXX: function not implemented yet @@ -295,12 +296,14 @@ static void remove_branch(KeyBindingTree *first) } -void OBBindings::removeAll() +void OBBindings::removeAllKeys() { + grabKeys(false); if (_keytree.first_child) { remove_branch(_keytree.first_child); _keytree.first_child = 0; } + grabKeys(true); } @@ -333,10 +336,10 @@ void OBBindings::grabKeys(bool grab) } -void OBBindings::fire(unsigned int modifiers, unsigned int key, Time time) +void OBBindings::fireKey(unsigned int modifiers, unsigned int key, Time time) { if (key == _resetkey.key && modifiers == _resetkey.modifiers) { - reset(this); + resetChains(this); } else { KeyBindingTree *p = _curpos->first_child; while (p) { @@ -353,7 +356,7 @@ void OBBindings::fire(unsigned int modifiers, unsigned int key, Time time) KeyData *data = new_key_data(win, time, modifiers, key); python_callback(p->callback, (PyObject*)data); Py_DECREF((PyObject*)data); - reset(this); + resetChains(this); } break; } @@ -362,7 +365,7 @@ void OBBindings::fire(unsigned int modifiers, unsigned int key, Time time) } } -void OBBindings::reset(OBBindings *self) +void OBBindings::resetChains(OBBindings *self) { self->_timer.stop(); self->grabKeys(false); @@ -399,7 +402,14 @@ bool OBBindings::addButton(const std::string &but, MouseContext context, bind->binding.key = b.key; bind->binding.modifiers = b.modifiers; _buttons[context].push_back(bind); - // XXX GRAB the new button everywhere! + // grab the button on all clients + for (int sn = 0; sn < Openbox::instance->screenCount(); ++sn) { + OBScreen *s = Openbox::instance->screen(sn); + OBScreen::ClientList::iterator c_it, c_end = s->clients.end(); + for (c_it = s->clients.begin(); c_it != c_end; ++c_it) { + grabButton(true, bind->binding, context, *c_it); + } + } } else bind = *it; Py_XDECREF(bind->callback[action]); // if it was already bound, unbind it @@ -408,41 +418,72 @@ bool OBBindings::addButton(const std::string &but, MouseContext context, return true; } +void OBBindings::removeAllButtons() +{ + for (int i = i; i < NUM_MOUSE_CONTEXT; ++i) { + ButtonBindingList::iterator it, end = _buttons[i].end(); + for (it = _buttons[i].begin(); it != end; ++it) { + for (int a = 0; a < NUM_MOUSE_ACTION; ++a) { + Py_XDECREF((*it)->callback[a]); + (*it)->callback[a] = 0; + } + // ungrab the button on all clients + for (int sn = 0; sn < Openbox::instance->screenCount(); ++sn) { + OBScreen *s = Openbox::instance->screen(sn); + OBScreen::ClientList::iterator c_it, c_end = s->clients.end(); + for (c_it = s->clients.begin(); c_it != c_end; ++c_it) { + grabButton(false, (*it)->binding, (MouseContext)i, *c_it); + } + } + } + } +} + +void OBBindings::grabButton(bool grab, const Binding &b, MouseContext context, + OBClient *client) +{ + Window win; + int mode = GrabModeAsync; + switch(context) { + case MC_Frame: + win = client->frame->window(); + break; + case MC_Window: + win = client->frame->plate(); + mode = GrabModeSync; // this is handled in fireButton + break; + default: + // any other elements already get button events, don't grab on them + return; + } + if (grab) + otk::OBDisplay::grabButton(b.key, b.modifiers, win, false, + ButtonPressMask | ButtonMotionMask | + ButtonReleaseMask, mode, GrabModeAsync, + None, None, false); + else + otk::OBDisplay::ungrabButton(b.key, b.modifiers, win); +} + void OBBindings::grabButtons(bool grab, OBClient *client) { for (int i = 0; i < NUM_MOUSE_CONTEXT; ++i) { - Window win; - int mode = GrabModeAsync; - switch (i) { - case MC_Frame: - win = client->frame->window(); - break; - case MC_Window: - win = client->frame->plate(); - mode = GrabModeSync; // this is handled in the plate's buttonPressHandler - break; - default: - // any other elements already get button events, don't grab on them - continue; - } ButtonBindingList::iterator it, end = _buttons[i].end(); for (it = _buttons[i].begin(); it != end; ++it) - if (grab) - otk::OBDisplay::grabButton((*it)->binding.key, - (*it)->binding.modifiers, win, false, - ButtonPressMask | ButtonMotionMask | - ButtonReleaseMask, mode, GrabModeAsync, - None, None, false); - else - otk::OBDisplay::ungrabButton((*it)->binding.key, - (*it)->binding.modifiers, win); + grabButton(grab, (*it)->binding, (MouseContext)i, client); } } -void OBBindings::fire(ButtonData *data) +void OBBindings::fireButton(ButtonData *data) { printf("but.mods %d.%d\n", data->button, data->state); + if (data->context == MC_Window) { + // these are grabbed in Sync mode to allow the press to be normal to the + // client + XAllowEvents(otk::OBDisplay::display, ReplayPointer, data->time); + } + ButtonBindingList::iterator it, end = _buttons[data->context].end(); for (it = _buttons[data->context].begin(); it != end; ++it) if ((*it)->binding.key == data->button &&