X-Git-Url: https://git.dogcows.com/gitweb?a=blobdiff_plain;f=util%2Fepist%2Fscreen.cc;h=db100889fb64e97b1182300f6d424494a16893c2;hb=e8fcc69d4b63f6d8531d93f61ead8eda8f45577d;hp=afe5b8d1e6b53e00f952dbceb45f7ca149ad1919;hpb=b4411cb1ef1c25a287181b570e974545e1010530;p=chaz%2Fopenbox diff --git a/util/epist/screen.cc b/util/epist/screen.cc index afe5b8d1..db100889 100644 --- a/util/epist/screen.cc +++ b/util/epist/screen.cc @@ -70,7 +70,9 @@ screen::screen(epist *epist, int number) { } XSelectInput(_epist->getXDisplay(), _root, PropertyChangeMask); - + + updateNumDesktops(); + updateActiveDesktop(); updateClientList(); updateActiveWindow(); } @@ -121,6 +123,10 @@ void screen::processEvent(const XEvent &e) { switch (e.type) { case PropertyNotify: // root window + if (e.xproperty.atom == _xatom->getAtom(XAtom::net_number_of_desktops)) + updateNumDesktops(); + if (e.xproperty.atom == _xatom->getAtom(XAtom::net_current_desktop)) + updateActiveDesktop(); if (e.xproperty.atom == _xatom->getAtom(XAtom::net_active_window)) updateActiveWindow(); if (e.xproperty.atom == _xatom->getAtom(XAtom::net_client_list)) { @@ -143,28 +149,95 @@ void screen::processEvent(const XEvent &e) { } void screen::handleKeypress(const XEvent &e) { - list::const_iterator it = _epist->actions().begin(); - list::const_iterator end = _epist->actions().end(); + ActionList::const_iterator it = _epist->actions().begin(); + ActionList::const_iterator end = _epist->actions().end(); for (; it != end; ++it) { if (e.xkey.keycode == it->keycode() && - e.xkey.state == it->modifierMask() ) - { + e.xkey.state == it->modifierMask()) { + switch (it->type()) { + case Action::nextWorkspace: + cycleWorkspace(true); + return; + + case Action::prevWorkspace: + cycleWorkspace(false); + return; + + case Action::nextWindow: + cycleWindow(true); + return; + + case Action::prevWindow: + cycleWindow(false); + return; + + case Action::nextWindowOnAllWorkspaces: + cycleWindow(true, true); + return; + + case Action::prevWindowOnAllWorkspaces: + cycleWindow(false, true); + return; + + case Action::nextWindowOfClass: + cycleWindow(true, false, true); + return; + + case Action::prevWindowOfClass: + cycleWindow(false, false, true); + return; + + case Action::nextWindowOfClassOnAllWorkspaces: + cycleWindow(true, true, true); + return; + + case Action::prevWindowOfClassOnAllWorkspaces: + cycleWindow(false, true, true); + return; + + case Action::changeWorkspace: + changeWorkspace(it->number()); + return; + } + + // these actions require an active window + if (_active != _clients.end()) { + XWindow *window = *_active; + switch (it->type()) { - case Action::nextWorkspace: - cycleWorkspace(true); - break; - case Action::prevWorkspace: - cycleWorkspace(false); - break; - case Action::changeWorkspace: - changeWorkspace(it->number()); - break; - case Action::shade: - toggleShaded((*_active)->window()); - break; + case Action::iconify: + window->iconify(); + return; + + case Action::close: + window->close(); + return; + + case Action::raise: + window->raise(); + return; + + case Action::lower: + window->lower(); + return; + + case Action::sendToWorkspace: + window->sendTo(it->number()); + return; + + case Action::toggleomnipresent: + if (window->desktop() == 0xffffffff) + window->sendTo(_active_desktop); + else + window->sendTo(0xffffffff); + return; + + case Action::toggleshade: + window->shade(! window->shaded()); + return; } - break; } + } } } @@ -185,6 +258,25 @@ bool screen::doAddWindow(Window window) const { } +void screen::updateNumDesktops() { + assert(_managed); + + if (! _xatom->getValue(_root, XAtom::net_number_of_desktops, XAtom::cardinal, + (unsigned long)_num_desktops)) + _num_desktops = 1; // assume that there is at least 1 desktop! +} + + +void screen::updateActiveDesktop() { + assert(_managed); + + if (! _xatom->getValue(_root, XAtom::net_current_desktop, XAtom::cardinal, + (unsigned long)_active_desktop)) + _active_desktop = 0; // there must be at least one desktop, and it must + // be the current one +} + + void screen::updateClientList() { assert(_managed); @@ -216,7 +308,8 @@ void screen::updateClientList() { if (it == end) { // didn't already exist if (doAddWindow(rootclients[i])) { cout << "Added window: 0x" << hex << rootclients[i] << dec << endl; - _clients.insert(insert_point, new XWindow(_epist, rootclients[i])); + _clients.insert(insert_point, new XWindow(_epist, this, + rootclients[i])); } } } @@ -265,34 +358,63 @@ void screen::updateActiveWindow() { } */ -void screen::cycleWorkspace(const bool forward) const { - unsigned long currentDesktop = 0; - unsigned long numDesktops = 0; - - if (_xatom->getValue(_root, XAtom::net_current_desktop, XAtom::cardinal, - currentDesktop)) { - if (forward) - ++currentDesktop; - else - --currentDesktop; - - _xatom->getValue(_root, XAtom::net_number_of_desktops, XAtom::cardinal, - numDesktops); + +void screen::cycleWindow(const bool forward, const bool alldesktops, + const bool sameclass) const { + assert(_managed); + + if (_clients.empty()) return; - if ( ( (signed)currentDesktop) == -1) - currentDesktop = numDesktops - 1; - else if (currentDesktop >= numDesktops) - currentDesktop = 0; + WindowList::const_iterator target = _active; + + if (target == _clients.end()) + target = _clients.begin(); + + do { + if (forward) { + ++target; + if (target == _clients.end()) + target = _clients.begin(); + } else { + if (target == _clients.begin()) + target = _clients.end(); + --target; + } + } while (target == _clients.end() || + (*target)->iconic() || + (! alldesktops && (*target)->desktop() != _active_desktop) || + (sameclass && _active != _clients.end() && + (*target)->appClass() != (*_active)->appClass())); + + if (target != _clients.end()) + (*target)->focus(); +} + - changeWorkspace(currentDesktop); +void screen::cycleWorkspace(const bool forward, const bool loop) const { + assert(_managed); + + unsigned int destination = _active_desktop; + + if (forward) { + if (destination < _num_desktops - 1) + ++destination; + else if (loop) + destination = 0; + } else { + if (destination > 0) + --destination; + else if (loop) + destination = _num_desktops - 1; } + + if (destination != _active_desktop) + changeWorkspace(destination); } + void screen::changeWorkspace(const int num) const { - _xatom->sendClientMessage(_root, XAtom::net_current_desktop, _root, num); -} + assert(_managed); -void screen::toggleShaded(const Window win) const { - _xatom->sendClientMessage(_root, XAtom::net_wm_state, win, 2, - XAtom::net_wm_state_shaded); + _xatom->sendClientMessage(_root, XAtom::net_current_desktop, _root, num); }