]> icculus.org git repositories - mikachu/openbox.git/blob - util/epist/epist.cc
add iconifying
[mikachu/openbox.git] / util / epist / epist.cc
1 // -*- mode: C++; indent-tabs-mode: nil; -*-
2 // epist.cc for Epistophy - a key handler for NETWM/EWMH window managers.
3 // Copyright (c) 2002 - 2002 Ben Jansens <ben at orodu.net>
4 //
5 // Permission is hereby granted, free of charge, to any person obtaining a
6 // copy of this software and associated documentation files (the "Software"),
7 // to deal in the Software without restriction, including without limitation
8 // the rights to use, copy, modify, merge, publish, distribute, sublicense,
9 // and/or sell copies of the Software, and to permit persons to whom the
10 // Software is furnished to do so, subject to the following conditions:
11 //
12 // The above copyright notice and this permission notice shall be included in
13 // all copies or substantial portions of the Software.
14 //
15 // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16 // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17 // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.  IN NO EVENT SHALL
18 // THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19 // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
20 // FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
21 // DEALINGS IN THE SOFTWARE.
22
23 #ifdef    HAVE_CONFIG_H
24 #  include "../../config.h"
25 #endif // HAVE_CONFIG_H
26
27 extern "C" {
28 #ifdef    HAVE_UNISTD_H
29 #  include <sys/types.h>
30 #  include <unistd.h>
31 #endif // HAVE_UNISTD_H
32
33 #ifdef    HAVE_STDLIB_H
34 #  include <stdlib.h>
35 #endif // HAVE_STDLIB_H
36
37 #ifdef    HAVE_SIGNAL_H
38 #  include <signal.h>
39 #endif // HAVE_SIGNAL_H
40
41 #ifdef    HAVE_LIBGEN_H
42 #  include <libgen.h>
43 #endif // HAVE_LIBGEN_H
44 }
45
46 #include <iostream>
47 #include <string>
48
49 using std::cout;
50 using std::endl;
51 using std::string;
52
53 #include "epist.hh"
54 #include "screen.hh"
55 #include "window.hh"
56 #include "../../src/XAtom.hh"
57
58
59 epist::epist(char **argv, char *dpy_name, char *rc_file)
60   : BaseDisplay(argv[0], dpy_name) {
61
62   _argv = argv;
63
64   if (rc_file)
65     _rc_file = rc_file;
66   else
67     _rc_file = expandTilde("~/.openbox/epistrc");
68
69   _xatom = new XAtom(getXDisplay());
70
71   screen *s = new screen(this, DefaultScreen(getXDisplay()));
72   if (s->managed())
73     _screens.push_back(s);
74   if (_screens.empty()) {
75     cout << "No compatible window manager found on any screens. Aborting.\n";
76     ::exit(1);
77   }
78
79   _actions.push_back(Action(Action::nextWorkspace,
80                             XKeysymToKeycode(getXDisplay(),
81                                              XStringToKeysym("Tab")),
82                             ControlMask));
83   _actions.push_back(Action(Action::prevWorkspace,
84                            XKeysymToKeycode(getXDisplay(),
85                                              XStringToKeysym("Tab")),
86                            ControlMask | ShiftMask));
87   _actions.push_back(Action(Action::toggleshade,
88                             XKeysymToKeycode(getXDisplay(),
89                                              XStringToKeysym("F5")),
90                             Mod1Mask));
91   _actions.push_back(Action(Action::close,
92                             XKeysymToKeycode(getXDisplay(),
93                                              XStringToKeysym("F4")),
94                             Mod1Mask));
95   _actions.push_back(Action(Action::nextWindow,
96                             XKeysymToKeycode(getXDisplay(),
97                                              XStringToKeysym("Tab")),
98                             Mod1Mask));
99   _actions.push_back(Action(Action::prevWindow,
100                            XKeysymToKeycode(getXDisplay(),
101                                              XStringToKeysym("Tab")),
102                            Mod1Mask | ShiftMask));
103   _actions.push_back(Action(Action::raise,
104                            XKeysymToKeycode(getXDisplay(),
105                                              XStringToKeysym("Up")),
106                            Mod1Mask));
107   _actions.push_back(Action(Action::lower,
108                            XKeysymToKeycode(getXDisplay(),
109                                              XStringToKeysym("Down")),
110                            Mod1Mask));
111   _actions.push_back(Action(Action::iconify,
112                            XKeysymToKeycode(getXDisplay(),
113                                              XStringToKeysym("I")),
114                            Mod1Mask | ControlMask));
115   activateGrabs();
116 }
117
118
119 epist::~epist() {
120   delete _xatom;
121 }
122
123 void epist::activateGrabs() {
124
125   ScreenList::const_iterator scrit, scrend = _screens.end();
126   
127   for (scrit = _screens.begin(); scrit != scrend; ++scrit) {
128     ActionList::const_iterator ait, end = _actions.end();
129
130     for(ait = _actions.begin(); ait != end; ++ait) {
131       XGrabKey(getXDisplay(), ait->keycode(), ait->modifierMask(),
132                (*scrit)->rootWindow(), False, GrabModeAsync, GrabModeAsync);
133     }
134   }
135 }
136
137
138 bool epist::handleSignal(int sig) {
139   switch (sig) {
140   case SIGHUP:
141     cout << "epist: Restarting on request.\n";
142     execvp(_argv[0], _argv);
143     execvp(basename(_argv[0]), _argv);
144     return false;  // this should be unreachable
145
146   case SIGTERM:
147   case SIGINT:
148   case SIGPIPE:
149     shutdown();
150     return true;
151   }
152
153   return false;
154 }
155
156
157 void epist::process_event(XEvent *e) {
158   Window root;
159
160   if (e->xany.type == KeyPress)
161     root = e->xkey.root;
162   else
163     root = e->xany.window;
164   
165   ScreenList::const_iterator it, end = _screens.end();
166   for (it = _screens.begin(); it != end; ++it) {
167     if ((*it)->rootWindow() == root) {
168       (*it)->processEvent(*e);
169       return;
170     }
171   }
172
173   // wasnt a root window, try for client windows
174   XWindow *w = findWindow(e->xany.window);
175   if (w) w->processEvent(*e);
176 }
177   
178
179 void epist::addWindow(XWindow *window) {
180   _windows.insert(WindowLookupPair(window->window(), window));
181 }
182
183
184 void epist::removeWindow(XWindow *window) {
185   _windows.erase(window->window());
186 }
187
188
189 XWindow *epist::findWindow(Window window) const {
190   WindowLookup::const_iterator it = _windows.find(window);
191   if (it != _windows.end())
192     return it->second;
193
194   return 0;
195 }