summaryrefslogtreecommitdiff
path: root/util/epist/epist.cc
diff options
context:
space:
mode:
authorDana Jansens <danakj@orodu.net>2002-07-11 18:13:38 +0000
committerDana Jansens <danakj@orodu.net>2002-07-11 18:13:38 +0000
commit0aae72418670c23f3fac604aca62cc8a461704f6 (patch)
tree656c469e6ce645039b1caa526edbaf19f92a2940 /util/epist/epist.cc
parent7aea20d6123a76acaf711d09e94961c979d19568 (diff)
epist now handles -display on the command line, and gets the client list and act
ive window
Diffstat (limited to 'util/epist/epist.cc')
-rw-r--r--util/epist/epist.cc179
1 files changed, 179 insertions, 0 deletions
diff --git a/util/epist/epist.cc b/util/epist/epist.cc
new file mode 100644
index 00000000..7bb16d93
--- /dev/null
+++ b/util/epist/epist.cc
@@ -0,0 +1,179 @@
+// -*- mode: C++; indent-tabs-mode: nil; -*-
+// epist.cc for Epistory - a key handler for NETWM/EWMH window managers.
+// Copyright (c) 2002 - 2002 Ben Jansens <ben at orodu.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.
+
+#ifdef HAVE_CONFIG_H
+# include "../../config.h"
+#endif // HAVE_CONFIG_H
+
+extern "C" {
+#ifdef HAVE_UNISTD_H
+# include <sys/types.h>
+# include <unistd.h>
+#endif // HAVE_UNISTD_H
+
+#ifdef HAVE_STDLIB_H
+# include <stdlib.h>
+#endif // HAVE_STDLIB_H
+
+#ifdef HAVE_SIGNAL_H
+# include <signal.h>
+#endif // HAVE_SIGNAL_H
+
+#ifdef HAVE_LIBGEN_H
+# include <libgen.h>
+#endif // HAVE_LIBGEN_H
+}
+
+#include <iostream>
+#include <string>
+
+using std::cout;
+using std::endl;
+using std::string;
+
+#include "epist.hh"
+#include "process.hh"
+#include "../../src/XAtom.hh"
+
+bool _shutdown = false;
+char **_argv;
+char *_display_name = 0;
+Display *_display = 0;
+Window _root = None;
+XAtom *_xatom;
+
+
+#ifdef HAVE_SIGACTION
+static void signalhandler(int sig)
+#else // HAVE_SIGACTION
+static RETSIGTYPE signalhandler(int sig)
+#endif // HAVE_SIGACTION
+{
+ switch (sig) {
+ case SIGSEGV:
+ cout << "epist: Segmentation fault. Aborting and dumping core.\n";
+ abort();
+ case SIGHUP:
+ cout << "epist: Restarting on request.\n";
+ execvp(_argv[0], _argv);
+ execvp(basename(_argv[0]), _argv);
+ }
+ _shutdown = true;
+
+#ifndef HAVE_SIGACTION
+ // assume broken, braindead sysv signal semantics
+ signal(sig, (RETSIGTYPE (*)(int)) signalhandler);
+#endif // HAVE_SIGACTION
+}
+
+
+void parseCommandLine(int argc, char **argv) {
+ _argv = argv;
+
+ for (int i = 1; i < argc; ++i) {
+ if (string(argv[i]) == "-display") {
+ if (++i >= argc) {
+ cout << "error:: '-display' requires an argument\n";
+ exit(1);
+ }
+ _display_name = argv[i];
+
+ string dtmp = (string)"DISPLAY=" + _display_name;
+ if (putenv(const_cast<char*>(dtmp.c_str()))) {
+ cout << "warning: couldn't set environment variable 'DISPLAY'\n";
+ perror("putenv()");
+ }
+ }
+ }
+}
+
+
+bool findSupportingWM() {
+ Window support_win;
+ if (! _xatom->getValue(_root, XAtom::net_supporting_wm_check, XAtom::window,
+ support_win) || support_win == None)
+ return false;
+
+ string title;
+ _xatom->getValue(support_win, XAtom::net_wm_name, XAtom::utf8, title);
+ cout << "Found compatible window manager: " << title << endl;
+ return true;
+}
+
+
+int main(int argc, char **argv) {
+#ifdef HAVE_SIGACTION
+ struct sigaction action;
+
+ action.sa_handler = signalhandler;
+ action.sa_mask = sigset_t();
+ action.sa_flags = SA_NOCLDSTOP | SA_NODEFER;
+
+ sigaction(SIGPIPE, &action, NULL);
+ sigaction(SIGSEGV, &action, NULL);
+ sigaction(SIGFPE, &action, NULL);
+ sigaction(SIGTERM, &action, NULL);
+ sigaction(SIGINT, &action, NULL);
+ sigaction(SIGHUP, &action, NULL);
+#else // !HAVE_SIGACTION
+ signal(SIGPIPE, (RETSIGTYPE (*)(int)) signalhandler);
+ signal(SIGSEGV, (RETSIGTYPE (*)(int)) signalhandler);
+ signal(SIGFPE, (RETSIGTYPE (*)(int)) signalhandler);
+ signal(SIGTERM, (RETSIGTYPE (*)(int)) signalhandler);
+ signal(SIGINT, (RETSIGTYPE (*)(int)) signalhandler);
+ signal(SIGHUP, (RETSIGTYPE (*)(int)) signalhandler);
+#endif // HAVE_SIGACTION
+
+ parseCommandLine(argc, argv);
+
+ _display = XOpenDisplay(_display_name);
+ if (! _display) {
+ cout << "Connection to X server '" << _display_name << "' failed.\n";
+ return 1;
+ }
+ _root = RootWindow(_display, DefaultScreen(_display));
+ _xatom = new XAtom(_display);
+
+ XSelectInput(_display, _root, PropertyChangeMask);
+
+ // find a window manager supporting NETWM, waiting for it to load if we must
+ while (! (_shutdown || findSupportingWM()));
+
+ if (! _shutdown) {
+ updateClientList();
+ updateActiveWindow();
+ }
+
+ while (! _shutdown) {
+ if (XPending(_display)) {
+ XEvent e;
+ XNextEvent(_display, &e);
+ processEvent(e);
+ } else {
+ usleep(300);
+ }
+ }
+
+ delete _xatom;
+ XCloseDisplay(_display);
+ return 0;
+}