summaryrefslogtreecommitdiffstats
path: root/src/util.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'src/util.cpp')
-rw-r--r--src/util.cpp324
1 files changed, 324 insertions, 0 deletions
diff --git a/src/util.cpp b/src/util.cpp
new file mode 100644
index 0000000..6ddf920
--- /dev/null
+++ b/src/util.cpp
@@ -0,0 +1,324 @@
+/*
+ * Copyright (C) 2004 Girish Ramakrishnan All Rights Reserved.
+ *
+ * This is free software; you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation; either version 2 of the License, or
+ * (at your option) any later version.
+ *
+ * This software is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this software; if not, write to the Free Software
+ * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307,
+ * USA.
+ */
+
+// $Id: util.cpp,v 1.7 2005/01/29 09:56:08 cs19713 Exp $
+
+#include "trace.h"
+#include "util.h"
+
+#include <X11/Xutil.h>
+#include <Xmu/WinUtil.h>
+#include <X11/Xatom.h>
+#include <X11/cursorfont.h>
+
+#include <stdio.h>
+
+/*
+ * Assert validity of the window id. Get window attributes for the heck of it
+ * and see if the request went through.
+ */
+bool isValidWindowId(Display *display, Window w)
+{
+ XWindowAttributes attrib;
+ return XGetWindowAttributes(display, w, &attrib) != 0;
+}
+
+/*
+ * Checks if this window is a normal window (i.e)
+ * - Has a WM_STATE
+ * - Not modal window
+ * - Not a purely transient window (with no window type set)
+ * - Not a special window (desktop/menu/util) as indicated in the window type
+ */
+bool isNormalWindow(Display *display, Window w)
+{
+ Atom __attribute__ ((unused)) type;
+ int __attribute__ ((unused)) format;
+ unsigned long __attribute__ ((unused)) left;
+ Atom *data = NULL;
+ unsigned long nitems;
+ Window transient_for = None;
+
+ static Atom wmState = XInternAtom(display, "WM_STATE", False);
+ static Atom windowState = XInternAtom(display, "_NET_WM_STATE", False);
+ static Atom modalWindow =
+ XInternAtom(display, "_NET_WM_STATE_MODAL", False);
+
+ static Atom windowType = XInternAtom(display, "_NET_WM_WINDOW_TYPE", False);
+ static Atom normalWindow =
+ XInternAtom(display, "_NET_WM_WINDOW_TYPE_NORMAL", False);
+ static Atom dialogWindow =
+ XInternAtom(display, "_NET_WM_WINDOW_TYPE_DIALOG", False);
+
+ int ret = XGetWindowProperty(display, w, wmState, 0,
+ 10, False, AnyPropertyType, &type, &format,
+ &nitems, &left, (unsigned char **) &data);
+
+ TRACE("0x%x (%i)", (unsigned) w, (unsigned) w);
+
+ if (ret != Success || data == NULL) return false;
+ TRACE("\tHas WM_STATE");
+ if (data) XFree(data);
+
+ ret = XGetWindowProperty(display, w, windowState, 0,
+ 10, False, AnyPropertyType, &type, &format,
+ &nitems, &left, (unsigned char **) &data);
+ if (ret == Success)
+ {
+ unsigned int i;
+ for (i = 0; i < nitems; i++)
+ {
+ if (data[i] == modalWindow) break;
+ }
+ XFree(data);
+ if (i < nitems) { TRACE("\tMODAL"); return false; }
+ }
+ else TRACE("\tNo _NET_WM_STATE");
+
+ XGetTransientForHint(display, w, &transient_for);
+ TRACE("\tTransientFor=0x%x", (unsigned) transient_for);
+
+ ret = XGetWindowProperty(display, w, windowType, 0,
+ 10, False, AnyPropertyType, &type, &format,
+ &nitems, &left, (unsigned char **) &data);
+
+ if ((ret == Success) && data)
+ {
+ unsigned int i;
+ for (i = 0; i < nitems; i++)
+ {
+ if (data[i] != normalWindow && data[i] != dialogWindow) break;
+ }
+ XFree(data);
+ TRACE("\tAbnormal = %i", (i == nitems));
+ return (i == nitems);
+ }
+ else
+ return (transient_for == None);
+}
+
+/*
+ * Returns the contents of the _NET_WM_PID (which is supposed to contain the
+ * process id of the application that created the window)
+ */
+pid_t pid(Display *display, Window w)
+{
+ Atom actual_type;
+ int actual_format;
+ unsigned long nitems, leftover;
+ unsigned char *pid;
+ pid_t pid_return = -1;
+
+ if (XGetWindowProperty(display, w,
+ XInternAtom(display, "_NET_WM_PID", False), 0,
+ 1, False, XA_CARDINAL, &actual_type,
+ &actual_format, &nitems, &leftover, &pid) == Success)
+ {
+ if (pid) pid_return = *(pid_t *) pid;
+ XFree(pid);
+ }
+ TRACE("0x%x has PID=%i", (unsigned) w, pid_return);
+ return pid_return;
+}
+
+/*
+ * Sends ClientMessage to a window
+ */
+void sendMessage(Display* display, Window to, Window w, char* type,
+ int format, long mask, void* data, int size)
+{
+ XEvent ev;
+ memset(&ev, 0, sizeof(ev));
+ ev.xclient.type = ClientMessage;
+ ev.xclient.window = w;
+ ev.xclient.message_type = XInternAtom(display, type, True);
+ ev.xclient.format = format;
+ memcpy((char *) &ev.xclient.data, (const char *) data, size);
+ XSendEvent(display, to, False, mask, &ev);
+ XSync(display, False);
+}
+
+/*
+ * The Grand Window Analyzer. Checks if window w has a expected pid of epid
+ * or a expected name of ename
+ */
+bool analyzeWindow(Display *display, Window w, pid_t epid, const QString &ename)
+{
+ XClassHint ch;
+ pid_t apid = pid(display, w);
+
+ TRACE("WID=0x%x PID=%i ExpectedName=%s", (unsigned) w, (unsigned) epid,
+ ename.latin1());
+ if (epid == apid) return true;
+
+ // no plans to analyze windows without a name
+ char *window_name = NULL;
+ if (!XFetchName(display, w, &window_name)) return false;
+ TRACE("Window has name [%s]", window_name);
+ if (window_name) XFree(window_name); else return false;
+
+ bool this_is_our_man = false;
+ // lets try the program name
+ if (XGetClassHint(display, w, &ch))
+ {
+ if (QString(ch.res_name).find(ename, 0, FALSE) != -1)
+ {
+ TRACE("ResName [%s] matched", ch.res_name);
+ this_is_our_man = true;
+ }
+ else if (QString(ch.res_class).find(ename, 0, FALSE) != -1)
+ {
+ TRACE("ResClass [%s] matched", ch.res_class);
+ this_is_our_man = true;
+ }
+ else
+ {
+ // sheer desperation
+ char *wm_name = NULL;
+ XFetchName(display, w, &wm_name);
+ if (wm_name && (QString(wm_name).find(ename, 0, FALSE) != -1))
+ {
+ TRACE("WM_NAME [%s] matched", wm_name);
+ this_is_our_man = true;
+ }
+ }
+
+ if (ch.res_class) XFree(ch.res_class);
+ if (ch.res_name) XFree(ch.res_name);
+ }
+
+ // its probably a good idea to check (obsolete) WM_COMMAND here
+ return this_is_our_man;
+}
+
+Window activeWindow(Display *display)
+{
+ Atom active_window_atom = XInternAtom(display, "_NET_ACTIVE_WINDOW", True);
+ Atom type = None;
+ int format;
+ unsigned long nitems, after;
+ unsigned char *data = NULL;
+ int screen = DefaultScreen(display);
+ Window root = RootWindow(display, screen);
+ int r = XGetWindowProperty(display, root, active_window_atom,0, 1,
+ False, AnyPropertyType, &type, &format, &nitems, &after, &data);
+
+ Window w = None;
+ if ((r == Success) && data && (*(Window *)data != None))
+ w = *(Window *)data;
+ else
+ {
+ int revert;
+ XGetInputFocus(display, &w, &revert);
+ }
+ TRACE("Active window is 0x%x", (unsigned) w);
+ return w;
+}
+
+/*
+ * Requests user to select a window by grabbing the mouse. A left click will
+ * select the application. Clicking any other button will abort the selection
+ */
+Window selectWindow(Display *display, const char **err)
+{
+ int screen = DefaultScreen(display);
+ Window root = RootWindow(display, screen);
+
+ if (err) *err = NULL;
+
+ Cursor cursor = XCreateFontCursor(display, XC_draped_box);
+ if (cursor == None)
+ {
+ if (err) *err = "Failed to create XC_draped_box";
+ return None;
+ }
+
+ if (XGrabPointer(display, root, False, ButtonPressMask | ButtonReleaseMask,
+ GrabModeSync, GrabModeAsync, None, cursor, CurrentTime)
+ != GrabSuccess)
+ {
+ if (err) *err = "Failed to grab mouse";
+ return None;
+ }
+
+ XAllowEvents(display, SyncPointer, CurrentTime);
+ XEvent event;
+ XWindowEvent(display, root, ButtonPressMask, &event);
+ Window selected_window =
+ (event.xbutton.subwindow == None) ? RootWindow(display, screen)
+ : event.xbutton.subwindow;
+ XUngrabPointer(display, CurrentTime);
+ XFreeCursor(display, cursor);
+
+ if (event.xbutton.button != Button1) return None;
+ return XmuClientWindow(display, selected_window);
+}
+
+void subscribe(Display *display, Window w, long mask, bool set)
+{
+ Window root = RootWindow(display, DefaultScreen(display));
+ XWindowAttributes attr;
+
+ TRACE("Window=0x%x Mask=%ld Set=%i", (unsigned) w, mask, set);
+
+ XGetWindowAttributes(display, w == None ? root : w, &attr);
+
+ if (set && (attr.your_event_mask & mask == mask)) return;
+ if (!set && (attr.your_event_mask | mask == attr.your_event_mask)) return;
+
+ XSelectInput(display, w == None ? root : w,
+ set ? attr.your_event_mask | mask : attr.your_event_mask & mask);
+ XSync(display, False);
+}
+
+// Returns the state of the SystemTray and the Wid if it exists
+SysTrayState sysTrayStatus(Display *display, Window *tray)
+{
+ Screen *screen = XDefaultScreenOfDisplay(display);
+ Window sys_tray;
+ Atom selection = None;
+
+ char temp[50];
+ sprintf(temp, "_NET_SYSTEM_TRAY_S%i", XScreenNumberOfScreen(screen));
+ selection = XInternAtom(display, temp, True);
+ if (selection == None) return SysTrayAbsent;
+ sys_tray = XGetSelectionOwner(display, selection);
+ if (tray) *tray = sys_tray;
+ return sys_tray == None ? SysTrayHidden : SysTrayPresent;
+}
+
+bool getCardinalProperty(Display *display, Window w, Atom prop, long *data)
+{
+ Atom type;
+ int format;
+ unsigned long nitems, bytes;
+ unsigned char *d = NULL;
+
+ if (XGetWindowProperty(display, w, prop, 0, 1, False, XA_CARDINAL,&type,
+ &format, &nitems, &bytes, &d) == Success && d)
+ {
+ TRACE("%ld", *(long *)d);
+ if (data) *data = *(long *)d;
+ XFree(d);
+ return true;
+ }
+ TRACE("FAILED");
+ return false;
+}
+