f65184802d
* NEW: Update copyright and license string in all source files. git-svn-id: svn://localhost/gambas/trunk@2241 867c0c6c-44f3-4631-809d-bfa615b0a4ec
689 lines
16 KiB
C
689 lines
16 KiB
C
/***************************************************************************
|
|
|
|
desktop.c
|
|
|
|
(c) 2000-2009 Benoît Minisini <gambas@users.sourceforge.net>
|
|
|
|
This program 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, or (at your option)
|
|
any later version.
|
|
|
|
This program 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 program; if not, write to the Free Software
|
|
Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
|
|
|
|
***************************************************************************/
|
|
|
|
#define __DESKTOP_C
|
|
|
|
#include "x11.h"
|
|
#include "desktop.h"
|
|
|
|
#if 0
|
|
typedef struct PROPERTY_FORMAT
|
|
{
|
|
const char *name;
|
|
Atom atom;
|
|
const char *format;
|
|
};
|
|
|
|
static PROPERTY_FORMAT _property_format[] = {
|
|
{"ARC" XA_ARC, "16iiccii" },
|
|
{"ATOM", XA_ATOM, "32a" },
|
|
{"BITMAP", XA_BITMAP, "32x" },
|
|
{"CARDINAL", XA_CARDINAL, "0c" },
|
|
{"COLORMAP", XA_COLORMAP, "32x" },
|
|
{"CURSOR", XA_CURSOR, "32x" },
|
|
{"DRAWABLE", XA_DRAWABLE, "32x" },
|
|
{"FONT", XA_FONT, "32x" },
|
|
{"INTEGER", XA_INTEGER, "0i" },
|
|
{"PIXMAP", XA_PIXMAP, "32x" },
|
|
{"POINT", XA_POINT, "16ii" },
|
|
{"RECTANGLE", XA_RECTANGLE, "16iicc" },
|
|
{"RGB_COLOR_MAP", XA_RGB_COLOR_MAP, "32xcccccccxx" },
|
|
{"STRING", XA_STRING, "8s" },
|
|
{"WINDOW", XA_WINDOW, "32x" },
|
|
{"VISUALID", XA_VISUALID, "32x" },
|
|
{"WM_COLORMAP_WINDOWS", 0, "32x" },
|
|
{"WM_COMMAND", XA_WM_COMMAND, "8s" },
|
|
{"WM_HINTS", XA_WM_HINTS, "32mbcxxiixx" },
|
|
{"WM_ICON_NAME", XA_WM_ICON_NAME, "8t" },
|
|
{"WM_ICON_SIZE", XA_WM_ICON_SIZE, "32cccccc" },
|
|
{"WM_NAME", XA_WM_NAME, "8t" },
|
|
{"WM_PROTOCOLS", 0, "32a" },
|
|
{"WM_SIZE_HINTS", XA_WM_SIZE_HINTS, "32mii" },
|
|
{"WM_STATE", 0, "32cx" }
|
|
};
|
|
#endif
|
|
|
|
DECLARE_EVENT(EVENT_Change);
|
|
DECLARE_EVENT(EVENT_Window);
|
|
|
|
/*BEGIN_METHOD_VOID(CDESKTOP_init)
|
|
|
|
X11_init();
|
|
|
|
END_METHOD*/
|
|
|
|
BEGIN_METHOD(CDESKTOP_find, GB_STRING title; GB_STRING klass; GB_STRING role)
|
|
|
|
Window *windows;
|
|
Window win;
|
|
int count;
|
|
int i;
|
|
char *title = MISSING(title) ? NULL : STRING(title);
|
|
int ltitle = MISSING(title) ? 0 : LENGTH(title);
|
|
char *klass = MISSING(klass) ? NULL : STRING(klass);
|
|
int lklass = MISSING(klass) ? 0 : LENGTH(klass);
|
|
char *role = MISSING(role) ? NULL : STRING(role);
|
|
int lrole = MISSING(role) ? 0 : LENGTH(role);
|
|
char *prop;
|
|
int lprop;
|
|
GB_ARRAY result;
|
|
|
|
if (X11_init())
|
|
return;
|
|
|
|
GB.Array.New(&result, GB_T_INTEGER, 0);
|
|
|
|
X11_find_windows(&windows, &count);
|
|
|
|
for (i = 0; i < count; i++)
|
|
{
|
|
win = windows[i];
|
|
//qDebug("win = %08X", win);
|
|
if (ltitle)
|
|
{
|
|
X11_get_window_title(win, &prop, &lprop);
|
|
//qDebug("title = %.*s", lprop, prop);
|
|
if (!GB.MatchString(title, ltitle, prop, lprop))
|
|
continue;
|
|
}
|
|
|
|
if (lklass)
|
|
{
|
|
X11_get_window_class(win, &prop, &lprop);
|
|
//qDebug("class = %.*s", lprop, prop);
|
|
if (!GB.MatchString(klass, lklass, prop, lprop))
|
|
continue;
|
|
}
|
|
|
|
if (lrole)
|
|
{
|
|
X11_get_window_role(win, &prop, &lprop);
|
|
//qDebug("role = %.*s", lprop, prop);
|
|
if (!GB.MatchString(role, lrole, prop, lprop))
|
|
continue;
|
|
}
|
|
|
|
*((int *)GB.Array.Add(result)) = (int)win;
|
|
}
|
|
|
|
GB.ReturnObject(result);
|
|
|
|
END_METHOD
|
|
|
|
|
|
BEGIN_METHOD(CDESKTOP_sendkey, GB_STRING key; GB_BOOLEAN press)
|
|
|
|
char *error;
|
|
|
|
if (X11_init())
|
|
return;
|
|
|
|
error = X11_send_key(GB.ToZeroString(ARG(key)), VARG(press));
|
|
if (error) GB.Error(error);
|
|
|
|
END_METHOD
|
|
|
|
|
|
BEGIN_PROPERTY(CDESKTOP_root)
|
|
|
|
if (X11_init())
|
|
return;
|
|
|
|
GB.ReturnInteger(X11_root);
|
|
|
|
END_PROPERTY
|
|
|
|
|
|
BEGIN_METHOD(CDESKTOP_get_window_property, GB_STRING name; GB_INTEGER window)
|
|
|
|
char *value;
|
|
Atom type;
|
|
Atom prop;
|
|
int format;
|
|
GB_ARRAY array;
|
|
int i, count;
|
|
char *name;
|
|
Window window;
|
|
|
|
if (X11_init())
|
|
return;
|
|
|
|
prop = X11_intern_atom(GB.ToZeroString(ARG(name)), FALSE);
|
|
if (prop == None)
|
|
{
|
|
GB.ReturnNull();
|
|
return;
|
|
}
|
|
|
|
window = VARGOPT(window, X11_root);
|
|
|
|
value = X11_get_property(window, prop, &type, &format, &count);
|
|
if (!value)
|
|
{
|
|
GB.ReturnNull();
|
|
return;
|
|
}
|
|
|
|
/*name = XGetAtomName(X11_display, type);
|
|
fprintf(stderr, "type = %s\n", name);
|
|
XFree(name);*/
|
|
|
|
if (type == XA_ATOM)
|
|
{
|
|
GB.Array.New(&array, GB_T_STRING, count);
|
|
for (i = 0; i < count; i++)
|
|
{
|
|
name = XGetAtomName(X11_display, *((Atom *)value + i));
|
|
GB.NewString((char **)GB.Array.Get(array, i), name, 0);
|
|
XFree(name);
|
|
}
|
|
GB.ReturnObject(array);
|
|
}
|
|
else if (type == X11_UTF8_STRING && format == 8)
|
|
{
|
|
count = GB.StringLength(value);
|
|
GB.Array.New(&array, GB_T_STRING, 0);
|
|
while (count > 0)
|
|
{
|
|
for (i = 0; i < count; i++)
|
|
{
|
|
if (!value[i])
|
|
break;
|
|
}
|
|
if (i)
|
|
GB.NewString((char **)GB.Array.Add(array), value, i);
|
|
i++;
|
|
value += i;
|
|
count -= i;
|
|
}
|
|
GB.ReturnObject(array);
|
|
}
|
|
else
|
|
{
|
|
switch(format)
|
|
{
|
|
case 16:
|
|
count = GB.StringLength(value) / sizeof(int);
|
|
GB.Array.New(&array, GB_T_SHORT, count);
|
|
for (i = 0; i < count; i++)
|
|
*((short *)GB.Array.Get(array, i)) = *((short *)value + i);
|
|
GB.ReturnObject(array);
|
|
break;
|
|
|
|
case 32: // A "long", not necessarily 32 bits!!
|
|
count = GB.StringLength(value) / sizeof(long);
|
|
GB.Array.New(&array, GB_T_INTEGER, count);
|
|
for (i = 0; i < count; i++)
|
|
*((int *)GB.Array.Get(array, i)) = *((long *)value + i);
|
|
GB.ReturnObject(array);
|
|
break;
|
|
|
|
default:
|
|
GB.ReturnString(value);
|
|
}
|
|
}
|
|
|
|
END_METHOD
|
|
|
|
BEGIN_METHOD(CDESKTOP_set_window_property, GB_STRING name; GB_STRING type; GB_VARIANT value; GB_INTEGER window)
|
|
|
|
Atom type;
|
|
Atom prop;
|
|
int format;
|
|
int count;
|
|
void *data;
|
|
void *object;
|
|
Window window;
|
|
void *buffer = NULL;
|
|
#if OS_64BITS
|
|
long padded_value;
|
|
long *padded_data;
|
|
int i;
|
|
#endif
|
|
|
|
if (X11_init())
|
|
return;
|
|
|
|
prop = X11_intern_atom(GB.ToZeroString(ARG(name)), TRUE);
|
|
type = X11_intern_atom(GB.ToZeroString(ARG(type)), TRUE);
|
|
count = 0;
|
|
format = 0;
|
|
|
|
window = VARGOPT(window, X11_root);
|
|
|
|
switch(VARG(value).type)
|
|
{
|
|
case GB_T_STRING:
|
|
case GB_T_CSTRING:
|
|
format = 8;
|
|
data = VARG(value)._string.value;
|
|
count = GB.StringLength(data);
|
|
break;
|
|
|
|
case GB_T_BOOLEAN:
|
|
case GB_T_BYTE:
|
|
case GB_T_SHORT:
|
|
case GB_T_INTEGER:
|
|
format = 32;
|
|
#if OS_64BITS
|
|
padded_data = VARG(value)._integer.value;
|
|
data = &padded_data;
|
|
#else
|
|
data = &VARG(value)._integer.value;
|
|
#endif
|
|
count = 1;
|
|
break;
|
|
|
|
default:
|
|
if (VARG(value).type >= GB_T_OBJECT)
|
|
{
|
|
object = VARG(value)._object.value;
|
|
if (GB.Is(object, GB.FindClass("Array")))
|
|
{
|
|
data = GB.Array.Get((GB_ARRAY)object, 0);
|
|
count = GB.Array.Count((GB_ARRAY)object);
|
|
|
|
switch((int)GB.Array.Type(object))
|
|
{
|
|
case GB_T_BYTE:
|
|
format = 8;
|
|
break;
|
|
|
|
case GB_T_SHORT:
|
|
format = 16;
|
|
break;
|
|
|
|
case GB_T_INTEGER:
|
|
format = 32;
|
|
#if OS_64BITS
|
|
padded_data = (long *)alloca(sizeof(long) * count);
|
|
for (i = 0; i < count; i++)
|
|
padded_data[i] = *((long *)data + i);
|
|
data = padded_data;
|
|
#endif
|
|
break;
|
|
|
|
case GB_T_STRING:
|
|
if (type == X11_UTF8_STRING)
|
|
{
|
|
int i, len;
|
|
char *p;
|
|
char **d = (char **)data;
|
|
|
|
len = 0;
|
|
for (i = 0; i < count; i++)
|
|
len += GB.StringLength(d[i]) + 1;
|
|
|
|
GB.Alloc(&buffer, len);
|
|
|
|
format = 8;
|
|
data = buffer;
|
|
count = len;
|
|
|
|
p = buffer;
|
|
for (i = 0; i < count; i++)
|
|
{
|
|
len = GB.StringLength(d[i]);
|
|
memcpy(p, d[i], len + 1);
|
|
p += len + 1;
|
|
}
|
|
}
|
|
default:
|
|
goto __ERROR;
|
|
}
|
|
}
|
|
}
|
|
else
|
|
goto __ERROR;
|
|
|
|
break;
|
|
}
|
|
|
|
if (format)
|
|
X11_set_property(window, prop, type, format, data, count);
|
|
|
|
if (buffer)
|
|
GB.Free(&buffer);
|
|
|
|
return;
|
|
|
|
__ERROR:
|
|
|
|
GB.Error("Cannot deal with that datatype");
|
|
|
|
END_METHOD
|
|
|
|
BEGIN_METHOD(CDESKTOP_intern_atom, GB_STRING atom; GB_BOOLEAN create)
|
|
|
|
if (X11_init())
|
|
return;
|
|
|
|
GB.ReturnInteger(X11_intern_atom(GB.ToZeroString(ARG(atom)), !VARGOPT(create, FALSE)));
|
|
|
|
END_METHOD
|
|
|
|
BEGIN_METHOD(CDESKTOP_get_atom_name, GB_INTEGER atom)
|
|
|
|
char *name;
|
|
|
|
if (X11_init())
|
|
return;
|
|
|
|
name = XGetAtomName(X11_display, VARG(atom));
|
|
GB.ReturnNewZeroString(name);
|
|
XFree(name);
|
|
|
|
END_METHOD
|
|
|
|
BEGIN_METHOD(CDESKTOP_send_client_message, GB_STRING message; GB_OBJECT data; GB_INTEGER window)
|
|
|
|
GB_ARRAY array;
|
|
char *data = NULL;
|
|
int count = 0;
|
|
int format = 0;
|
|
#if OS_64BITS
|
|
long *padded_data;
|
|
int i;
|
|
#endif
|
|
|
|
if (X11_init())
|
|
return;
|
|
|
|
if (!MISSING(data) && VARG(data))
|
|
{
|
|
array = VARG(data);
|
|
data = GB.Array.Get(array, 0);
|
|
count = GB.Array.Count(array);
|
|
|
|
switch((int)GB.Array.Type(array))
|
|
{
|
|
case GB_T_BYTE:
|
|
format = 8;
|
|
break;
|
|
|
|
case GB_T_SHORT:
|
|
format = 16;
|
|
break;
|
|
|
|
case GB_T_INTEGER:
|
|
format = 32;
|
|
#if OS_64BITS
|
|
padded_data = (long *)alloca(sizeof(long) * count);
|
|
for (i = 0; i < count; i++)
|
|
padded_data[i] = *((long *)data + i);
|
|
data = padded_data;
|
|
#endif
|
|
break;
|
|
|
|
/*case GB_T_STRING:
|
|
format = 32;
|
|
// TODO: convert strings to Atom
|
|
break;*/
|
|
|
|
default:
|
|
return;
|
|
}
|
|
}
|
|
|
|
X11_send_client_message(X11_root, VARGOPT(window, X11_root),
|
|
X11_intern_atom(GB.ToZeroString(ARG(message)), TRUE),
|
|
data, format, count);
|
|
|
|
END_METHOD
|
|
|
|
BEGIN_PROPERTY(CDESKTOP_event_filter)
|
|
|
|
if (X11_init())
|
|
return;
|
|
|
|
if (READ_PROPERTY)
|
|
GB.ReturnBoolean(X11_event_filter_enabled);
|
|
else
|
|
X11_enable_event_filter(VPROP(GB_BOOLEAN));
|
|
|
|
END_PROPERTY
|
|
|
|
BEGIN_METHOD(CDESKTOP_watch_window, GB_INTEGER window; GB_BOOLEAN watch)
|
|
|
|
XWindowAttributes attr;
|
|
int mask = PropertyChangeMask | StructureNotifyMask;
|
|
|
|
if (X11_init())
|
|
return;
|
|
|
|
XGetWindowAttributes(X11_display, VARG(window), &attr);
|
|
|
|
if (VPROP(GB_BOOLEAN))
|
|
XSelectInput(X11_display, (Window)VARG(window), attr.your_event_mask | mask);
|
|
else
|
|
XSelectInput(X11_display, (Window)VARG(window), attr.your_event_mask & ~mask);
|
|
|
|
END_METHOD
|
|
|
|
BEGIN_METHOD(CDESKTOP_get_window_geometry, GB_INTEGER window)
|
|
|
|
GB_ARRAY array;
|
|
int *data;
|
|
Window root;
|
|
uint ignore;
|
|
|
|
if (X11_init())
|
|
return;
|
|
|
|
GB.Array.New(&array, GB_T_INTEGER, 4);
|
|
data = (int *)GB.Array.Get(array, 0);
|
|
|
|
XGetGeometry(X11_display, VARG(window), &root, &data[0], &data[1], (uint *)&data[2], (uint *)&data[3], &ignore, &ignore);
|
|
GB.ReturnObject(array);
|
|
|
|
END_METHOD
|
|
|
|
BEGIN_METHOD(CDESKTOP_make_icon, GB_OBJECT data; GB_INTEGER width; GB_INTEGER height)
|
|
|
|
GB_ARRAY array;
|
|
int *data;
|
|
int count;
|
|
int size;
|
|
int width = VARGOPT(width, -1);
|
|
int height = VARGOPT(height, width);
|
|
int w, h;
|
|
|
|
array = VARG(data);
|
|
if (GB.CheckObject(array))
|
|
return;
|
|
|
|
data = (int *)GB.Array.Get(array, 0);
|
|
count = GB.Array.Count(array);
|
|
|
|
if (width < 0)
|
|
{
|
|
for(;;)
|
|
{
|
|
if (count < 2)
|
|
break;
|
|
w = data[0];
|
|
h = data[1];
|
|
if (w > width)
|
|
{
|
|
width = w;
|
|
height = h;
|
|
}
|
|
size = w * h + 2;
|
|
data += size;
|
|
count -= size;
|
|
}
|
|
|
|
data = (int *)GB.Array.Get(array, 0);
|
|
count = GB.Array.Count(array);
|
|
}
|
|
|
|
for(;;)
|
|
{
|
|
if (count < 2)
|
|
{
|
|
GB.ReturnNull();
|
|
return;
|
|
}
|
|
w = data[0];
|
|
h = data[1];
|
|
if (w == width && h == height)
|
|
break;
|
|
size = w * h + 2;
|
|
data += size;
|
|
count -= size;
|
|
}
|
|
|
|
GB.ReturnObject(IMAGE.Create(w, h, GB_IMAGE_BGRA, (unsigned char *)&data[2]));
|
|
|
|
END_METHOD
|
|
|
|
BEGIN_METHOD(CDESKTOP_minimize_window, GB_INTEGER window; GB_BOOLEAN minimized)
|
|
|
|
if (VARG(minimized))
|
|
{
|
|
long state = IconicState;
|
|
|
|
X11_send_client_message(X11_root, VARG(window),
|
|
X11_intern_atom("WM_CHANGE_STATE", TRUE),
|
|
(char *)&state, 32, 1);
|
|
}
|
|
else
|
|
{
|
|
XMapWindow(X11_display, VARG(window));
|
|
}
|
|
|
|
END_METHOD
|
|
|
|
GB_DESC CDesktopDesc[] =
|
|
{
|
|
GB_DECLARE("_Desktop", 0), GB_VIRTUAL_CLASS(),
|
|
|
|
//GB_STATIC_METHOD("Init", NULL, CDESKTOP_init, NULL),
|
|
|
|
GB_CONSTANT("CurrentTime", "i", CurrentTime),
|
|
|
|
GB_STATIC_METHOD("FindWindow", "Integer[]", CDESKTOP_find, "[(Title)s(Application)s(Role)s]"),
|
|
GB_STATIC_METHOD("SendKey", NULL, CDESKTOP_sendkey, "(Key)s(Press)b"),
|
|
GB_STATIC_PROPERTY_READ("RootWindow", "i", CDESKTOP_root),
|
|
GB_STATIC_METHOD("GetWindowProperty", "v", CDESKTOP_get_window_property, "(Property)s[(Window)i]"),
|
|
//GB_STATIC_METHOD("GetWindowPropertyType", "s", CDESKTOP_get_window_property_type, "(Window)i(Property)s"),
|
|
GB_STATIC_METHOD("SetWindowProperty", NULL, CDESKTOP_set_window_property, "(Property)s(Type)s(Value)v[(Window)i]"),
|
|
GB_STATIC_METHOD("InternAtom", "i", CDESKTOP_intern_atom, "(Atom)s[(Create)b]"),
|
|
GB_STATIC_METHOD("GetAtomName", "s", CDESKTOP_get_atom_name, "(Atom)i"),
|
|
GB_STATIC_METHOD("SendClientMessageToRootWindow", NULL, CDESKTOP_send_client_message, "(Message)s[(Data)Array;(Window)i]"),
|
|
//GB_STATIC_PROPERTY_SELF("Root", "._Desktop.Window"),
|
|
//GB_STATIC_METHOD("_get", "._Desktop.Window"),
|
|
GB_STATIC_PROPERTY("EventFilter", "b", CDESKTOP_event_filter),
|
|
GB_STATIC_METHOD("WatchWindow", NULL, CDESKTOP_watch_window, "(Window)i(Watch)b"),
|
|
GB_STATIC_METHOD("GetWindowGeometry", "Integer[]", CDESKTOP_get_window_geometry, "(Window)i"),
|
|
GB_STATIC_METHOD("MakeIcon", "Image", CDESKTOP_make_icon, "(Data)Array;[(Width)i(Height)h]"),
|
|
GB_STATIC_METHOD("MinimizeWindow", NULL, CDESKTOP_minimize_window, "(Window)i(Minimized)b"),
|
|
|
|
GB_END_DECLARE
|
|
};
|
|
|
|
/****************************************************************************
|
|
|
|
_DesktopWatcher
|
|
|
|
****************************************************************************/
|
|
|
|
static CDESKTOPWATCHER *_watcher_list = NULL;
|
|
|
|
static void x11_event_filter(XEvent *e)
|
|
{
|
|
CDESKTOPWATCHER *watcher;
|
|
|
|
if (e->type == PropertyNotify)
|
|
{
|
|
LIST_for_each(watcher, _watcher_list)
|
|
{
|
|
if (watcher->window && e->xany.window != watcher->window)
|
|
continue;
|
|
if (watcher->property && e->xproperty.atom != watcher->property)
|
|
continue;
|
|
|
|
GB.Raise(watcher, EVENT_Change, 2, GB_T_INTEGER, e->xany.window, GB_T_INTEGER, e->xproperty.atom);
|
|
}
|
|
}
|
|
else if (e->type == ConfigureNotify)
|
|
{
|
|
LIST_for_each(watcher, _watcher_list)
|
|
{
|
|
if (watcher->window && e->xany.window != watcher->window)
|
|
continue;
|
|
|
|
GB.Raise(watcher, EVENT_Window,
|
|
5, GB_T_INTEGER, e->xany.window,
|
|
GB_T_INTEGER, e->xconfigure.x,
|
|
GB_T_INTEGER, e->xconfigure.y,
|
|
GB_T_INTEGER, e->xconfigure.width,
|
|
GB_T_INTEGER, e->xconfigure.height);
|
|
}
|
|
}
|
|
}
|
|
|
|
static void enable_event_filter(bool enable)
|
|
{
|
|
void (*set_event_filter)(void *) = NULL;
|
|
|
|
GB.GetComponentInfo("SET_EVENT_FILTER", POINTER(&set_event_filter));
|
|
if (set_event_filter)
|
|
(*set_event_filter)(enable ? x11_event_filter : 0);
|
|
}
|
|
|
|
BEGIN_METHOD(CDESKTOPWATCHER_new, GB_INTEGER window; GB_STRING property)
|
|
|
|
if (X11_init())
|
|
return;
|
|
|
|
WATCHER->window = VARGOPT(window, 0);
|
|
WATCHER->property = MISSING(property) ? 0 : X11_intern_atom(GB.ToZeroString(ARG(property)), FALSE);
|
|
|
|
if (!_watcher_list)
|
|
enable_event_filter(TRUE);
|
|
|
|
LIST_insert(&_watcher_list, WATCHER, &WATCHER->list);
|
|
|
|
END_METHOD
|
|
|
|
BEGIN_METHOD_VOID(CDESKTOPWATCHER_free)
|
|
|
|
LIST_remove(&_watcher_list, WATCHER, &WATCHER->list);
|
|
if (!_watcher_list)
|
|
enable_event_filter(FALSE);
|
|
|
|
END_METHOD
|
|
|
|
GB_DESC CDesktopWatcherDesc[] =
|
|
{
|
|
GB_DECLARE("_DesktopWatcher", sizeof(CDESKTOPWATCHER)),
|
|
|
|
GB_METHOD("_new", NULL, CDESKTOPWATCHER_new, "[(Window)i(Property)s]"),
|
|
GB_METHOD("_free", NULL, CDESKTOPWATCHER_free, NULL),
|
|
|
|
GB_EVENT("Change", NULL, "(Window)i(Property)i", &EVENT_Change),
|
|
GB_EVENT("Window", NULL, "(Window)i(X)i(Y)i(Width)i(Height)i", &EVENT_Window),
|
|
|
|
GB_END_DECLARE
|
|
};
|