From d964a507c305b4a395534d87190d9baebad4dcd8 Mon Sep 17 00:00:00 2001 From: dan corson Date: Mon, 14 Jul 2008 20:00:32 -0700 Subject: [PATCH 01/14] adding -z to lower window --- main.c | 533 +++++++++++++++++++++++++++++---------------------------- 1 file changed, 268 insertions(+), 265 deletions(-) diff --git a/main.c b/main.c index 9dc1600..a5d1be8 100644 --- a/main.c +++ b/main.c @@ -264,7 +264,7 @@ int main (int argc, char **argv) { /* {{{ */ } } - while ((opt = getopt(argc, argv, "FGVvhlupidmxa:r:s:c:t:w:k:o:n:g:e:b:N:I:T:R:")) != -1) { + while ((opt = getopt(argc, argv, "FGVvhlupidmxa:r:s:c:t:w:k:o:n:g:e:b:z:N:I:T:R:")) != -1) { missing_option = 0; switch (opt) { case 'F': @@ -289,7 +289,7 @@ int main (int argc, char **argv) { /* {{{ */ case 'p': options.show_pid = 1; break; - case 'a': case 'c': case 'R': + case 'a': case 'c': case 'R': case 'z': options.param_window = optarg; action = opt; break; @@ -355,7 +355,7 @@ int main (int argc, char **argv) { /* {{{ */ case 'm': ret = wm_info(disp); break; - case 'a': case 'c': case 'R': + case 'a': case 'c': case 'R': case 'z': case 't': case 'e': case 'b': case 'N': case 'I': case 'T': if (! options.param_window) { fputs("No window was specified.\n", stderr); @@ -381,323 +381,323 @@ int main (int argc, char **argv) { /* {{{ */ ret = change_geometry(disp); break; } - + XCloseDisplay(disp); return ret; } /* }}} */ static void init_charset (void) {/*{{{*/ - const gchar *charset; /* unused */ - gchar *lang = getenv("LANG") ? g_ascii_strup(getenv("LANG"), -1) : NULL; - gchar *lc_ctype = getenv("LC_CTYPE") ? g_ascii_strup(getenv("LC_CTYPE"), -1) : NULL; - - /* this glib function doesn't work on my system ... */ - envir_utf8 = g_get_charset(&charset); - - /* ... therefore we will examine the environment variables */ - if (lc_ctype && (strstr(lc_ctype, "UTF8") || strstr(lc_ctype, "UTF-8"))) { - envir_utf8 = TRUE; - } - else if (lang && (strstr(lang, "UTF8") || strstr(lang, "UTF-8"))) { - envir_utf8 = TRUE; - } - - g_free(lang); - g_free(lc_ctype); - - if (options.force_utf8) { - envir_utf8 = TRUE; - } - p_verbose("envir_utf8: %d\n", envir_utf8); + const gchar *charset; /* unused */ + gchar *lang = getenv("LANG") ? g_ascii_strup(getenv("LANG"), -1) : NULL; + gchar *lc_ctype = getenv("LC_CTYPE") ? g_ascii_strup(getenv("LC_CTYPE"), -1) : NULL; + + /* this glib function doesn't work on my system ... */ + envir_utf8 = g_get_charset(&charset); + + /* ... therefore we will examine the environment variables */ + if (lc_ctype && (strstr(lc_ctype, "UTF8") || strstr(lc_ctype, "UTF-8"))) { + envir_utf8 = TRUE; + } + else if (lang && (strstr(lang, "UTF8") || strstr(lang, "UTF-8"))) { + envir_utf8 = TRUE; + } + + g_free(lang); + g_free(lc_ctype); + + if (options.force_utf8) { + envir_utf8 = TRUE; + } + p_verbose("envir_utf8: %d\n", envir_utf8); }/*}}}*/ static int client_msg(Display *disp, Window win, char *msg, /* {{{ */ - unsigned long data0, unsigned long data1, - unsigned long data2, unsigned long data3, - unsigned long data4) { - XEvent event; - long mask = SubstructureRedirectMask | SubstructureNotifyMask; - - event.xclient.type = ClientMessage; - event.xclient.serial = 0; - event.xclient.send_event = True; - event.xclient.message_type = XInternAtom(disp, msg, False); - event.xclient.window = win; - event.xclient.format = 32; - event.xclient.data.l[0] = data0; - event.xclient.data.l[1] = data1; - event.xclient.data.l[2] = data2; - event.xclient.data.l[3] = data3; - event.xclient.data.l[4] = data4; - - if (XSendEvent(disp, DefaultRootWindow(disp), False, mask, &event)) { - return EXIT_SUCCESS; - } - else { - fprintf(stderr, "Cannot send %s event.\n", msg); - return EXIT_FAILURE; - } + unsigned long data0, unsigned long data1, + unsigned long data2, unsigned long data3, + unsigned long data4) { + XEvent event; + long mask = SubstructureRedirectMask | SubstructureNotifyMask; + + event.xclient.type = ClientMessage; + event.xclient.serial = 0; + event.xclient.send_event = True; + event.xclient.message_type = XInternAtom(disp, msg, False); + event.xclient.window = win; + event.xclient.format = 32; + event.xclient.data.l[0] = data0; + event.xclient.data.l[1] = data1; + event.xclient.data.l[2] = data2; + event.xclient.data.l[3] = data3; + event.xclient.data.l[4] = data4; + + if (XSendEvent(disp, DefaultRootWindow(disp), False, mask, &event)) { + return EXIT_SUCCESS; + } + else { + fprintf(stderr, "Cannot send %s event.\n", msg); + return EXIT_FAILURE; + } }/*}}}*/ static gchar *get_output_str (gchar *str, gboolean is_utf8) {/*{{{*/ - gchar *out; - - if (str == NULL) { - return NULL; + gchar *out; + + if (str == NULL) { + return NULL; + } + + if (envir_utf8) { + if (is_utf8) { + out = g_strdup(str); } - - if (envir_utf8) { - if (is_utf8) { - out = g_strdup(str); - } - else { - if (! (out = g_locale_to_utf8(str, -1, NULL, NULL, NULL))) { - p_verbose("Cannot convert string from locale charset to UTF-8.\n"); - out = g_strdup(str); - } - } + else { + if (! (out = g_locale_to_utf8(str, -1, NULL, NULL, NULL))) { + p_verbose("Cannot convert string from locale charset to UTF-8.\n"); + out = g_strdup(str); + } + } + } + else { + if (is_utf8) { + if (! (out = g_locale_from_utf8(str, -1, NULL, NULL, NULL))) { + p_verbose("Cannot convert string from UTF-8 to locale charset.\n"); + out = g_strdup(str); + } } else { - if (is_utf8) { - if (! (out = g_locale_from_utf8(str, -1, NULL, NULL, NULL))) { - p_verbose("Cannot convert string from UTF-8 to locale charset.\n"); - out = g_strdup(str); - } - } - else { - out = g_strdup(str); - } + out = g_strdup(str); } + } - return out; + return out; }/*}}}*/ static int wm_info (Display *disp) {/*{{{*/ - Window *sup_window = NULL; - gchar *wm_name = NULL; - gchar *wm_class = NULL; - unsigned long *wm_pid = NULL; - unsigned long *showing_desktop = NULL; - gboolean name_is_utf8 = TRUE; - gchar *name_out; - gchar *class_out; - + Window *sup_window = NULL; + gchar *wm_name = NULL; + gchar *wm_class = NULL; + unsigned long *wm_pid = NULL; + unsigned long *showing_desktop = NULL; + gboolean name_is_utf8 = TRUE; + gchar *name_out; + gchar *class_out; + + if (! (sup_window = (Window *)get_property(disp, DefaultRootWindow(disp), + XA_WINDOW, "_NET_SUPPORTING_WM_CHECK", NULL))) { if (! (sup_window = (Window *)get_property(disp, DefaultRootWindow(disp), - XA_WINDOW, "_NET_SUPPORTING_WM_CHECK", NULL))) { - if (! (sup_window = (Window *)get_property(disp, DefaultRootWindow(disp), - XA_CARDINAL, "_WIN_SUPPORTING_WM_CHECK", NULL))) { - fputs("Cannot get window manager info properties.\n" - "(_NET_SUPPORTING_WM_CHECK or _WIN_SUPPORTING_WM_CHECK)\n", stderr); - return EXIT_FAILURE; - } + XA_CARDINAL, "_WIN_SUPPORTING_WM_CHECK", NULL))) { + fputs("Cannot get window manager info properties.\n" + "(_NET_SUPPORTING_WM_CHECK or _WIN_SUPPORTING_WM_CHECK)\n", stderr); + return EXIT_FAILURE; } + } - /* WM_NAME */ + /* WM_NAME */ + if (! (wm_name = get_property(disp, *sup_window, + XInternAtom(disp, "UTF8_STRING", False), "_NET_WM_NAME", NULL))) { + name_is_utf8 = FALSE; if (! (wm_name = get_property(disp, *sup_window, - XInternAtom(disp, "UTF8_STRING", False), "_NET_WM_NAME", NULL))) { - name_is_utf8 = FALSE; - if (! (wm_name = get_property(disp, *sup_window, - XA_STRING, "_NET_WM_NAME", NULL))) { - p_verbose("Cannot get name of the window manager (_NET_WM_NAME).\n"); - } + XA_STRING, "_NET_WM_NAME", NULL))) { + p_verbose("Cannot get name of the window manager (_NET_WM_NAME).\n"); } - name_out = get_output_str(wm_name, name_is_utf8); - - /* WM_CLASS */ - if (! (wm_class = get_property(disp, *sup_window, - XInternAtom(disp, "UTF8_STRING", False), "WM_CLASS", NULL))) { - name_is_utf8 = FALSE; - if (! (wm_class = get_property(disp, *sup_window, - XA_STRING, "WM_CLASS", NULL))) { - p_verbose("Cannot get class of the window manager (WM_CLASS).\n"); - } - } - class_out = get_output_str(wm_class, name_is_utf8); - + } + name_out = get_output_str(wm_name, name_is_utf8); - /* WM_PID */ - if (! (wm_pid = (unsigned long *)get_property(disp, *sup_window, - XA_CARDINAL, "_NET_WM_PID", NULL))) { - p_verbose("Cannot get pid of the window manager (_NET_WM_PID).\n"); - } - - /* _NET_SHOWING_DESKTOP */ - if (! (showing_desktop = (unsigned long *)get_property(disp, DefaultRootWindow(disp), - XA_CARDINAL, "_NET_SHOWING_DESKTOP", NULL))) { - p_verbose("Cannot get the _NET_SHOWING_DESKTOP property.\n"); - } - - /* print out the info */ - printf("Name: %s\n", name_out ? name_out : "N/A"); - printf("Class: %s\n", class_out ? class_out : "N/A"); - - if (wm_pid) { - printf("PID: %lu\n", *wm_pid); - } - else { - printf("PID: N/A\n"); - } - - if (showing_desktop) { - printf("Window manager's \"showing the desktop\" mode: %s\n", - *showing_desktop == 1 ? "ON" : "OFF"); - } - else { - printf("Window manager's \"showing the desktop\" mode: N/A\n"); - } - - g_free(name_out); - g_free(sup_window); - g_free(wm_name); - g_free(wm_class); - g_free(wm_pid); - g_free(showing_desktop); - - return EXIT_SUCCESS; + /* WM_CLASS */ + if (! (wm_class = get_property(disp, *sup_window, + XInternAtom(disp, "UTF8_STRING", False), "WM_CLASS", NULL))) { + name_is_utf8 = FALSE; + if (! (wm_class = get_property(disp, *sup_window, + XA_STRING, "WM_CLASS", NULL))) { + p_verbose("Cannot get class of the window manager (WM_CLASS).\n"); + } + } + class_out = get_output_str(wm_class, name_is_utf8); + + + /* WM_PID */ + if (! (wm_pid = (unsigned long *)get_property(disp, *sup_window, + XA_CARDINAL, "_NET_WM_PID", NULL))) { + p_verbose("Cannot get pid of the window manager (_NET_WM_PID).\n"); + } + + /* _NET_SHOWING_DESKTOP */ + if (! (showing_desktop = (unsigned long *)get_property(disp, DefaultRootWindow(disp), + XA_CARDINAL, "_NET_SHOWING_DESKTOP", NULL))) { + p_verbose("Cannot get the _NET_SHOWING_DESKTOP property.\n"); + } + + /* print out the info */ + printf("Name: %s\n", name_out ? name_out : "N/A"); + printf("Class: %s\n", class_out ? class_out : "N/A"); + + if (wm_pid) { + printf("PID: %lu\n", *wm_pid); + } + else { + printf("PID: N/A\n"); + } + + if (showing_desktop) { + printf("Window manager's \"showing the desktop\" mode: %s\n", + *showing_desktop == 1 ? "ON" : "OFF"); + } + else { + printf("Window manager's \"showing the desktop\" mode: N/A\n"); + } + + g_free(name_out); + g_free(sup_window); + g_free(wm_name); + g_free(wm_class); + g_free(wm_pid); + g_free(showing_desktop); + + return EXIT_SUCCESS; }/*}}}*/ static int showing_desktop (Display *disp) {/*{{{*/ - unsigned long state; - - if (strcmp(options.param, "on") == 0) { - state = 1; - } - else if (strcmp(options.param, "off") == 0) { - state = 0; - } - else { - fputs("The argument to the -k option must be either \"on\" or \"off\"\n", stderr); - return EXIT_FAILURE; - } - - return client_msg(disp, DefaultRootWindow(disp), "_NET_SHOWING_DESKTOP", - state, 0, 0, 0, 0); + unsigned long state; + + if (strcmp(options.param, "on") == 0) { + state = 1; + } + else if (strcmp(options.param, "off") == 0) { + state = 0; + } + else { + fputs("The argument to the -k option must be either \"on\" or \"off\"\n", stderr); + return EXIT_FAILURE; + } + + return client_msg(disp, DefaultRootWindow(disp), "_NET_SHOWING_DESKTOP", + state, 0, 0, 0, 0); }/*}}}*/ static int change_viewport (Display *disp) {/*{{{*/ - unsigned long x, y; - const char *argerr = "The -o option expects two integers separated with a comma.\n"; - - if (sscanf(options.param, "%lu,%lu", &x, &y) == 2) { - return client_msg(disp, DefaultRootWindow(disp), "_NET_DESKTOP_VIEWPORT", - x, y, 0, 0, 0); - } - else { - fputs(argerr, stderr); - return EXIT_FAILURE; - } + unsigned long x, y; + const char *argerr = "The -o option expects two integers separated with a comma.\n"; + + if (sscanf(options.param, "%lu,%lu", &x, &y) == 2) { + return client_msg(disp, DefaultRootWindow(disp), "_NET_DESKTOP_VIEWPORT", + x, y, 0, 0, 0); + } + else { + fputs(argerr, stderr); + return EXIT_FAILURE; + } }/*}}}*/ static int change_geometry (Display *disp) {/*{{{*/ - unsigned long x, y; - const char *argerr = "The -g option expects two integers separated with a comma.\n"; - - if (sscanf(options.param, "%lu,%lu", &x, &y) == 2) { - return client_msg(disp, DefaultRootWindow(disp), "_NET_DESKTOP_GEOMETRY", - x, y, 0, 0, 0); - } - else { - fputs(argerr, stderr); - return EXIT_FAILURE; - } + unsigned long x, y; + const char *argerr = "The -g option expects two integers separated with a comma.\n"; + + if (sscanf(options.param, "%lu,%lu", &x, &y) == 2) { + return client_msg(disp, DefaultRootWindow(disp), "_NET_DESKTOP_GEOMETRY", + x, y, 0, 0, 0); + } + else { + fputs(argerr, stderr); + return EXIT_FAILURE; + } }/*}}}*/ static int change_number_of_desktops (Display *disp) {/*{{{*/ - unsigned long n; + unsigned long n; - if (sscanf(options.param, "%lu", &n) != 1) { - fputs("The -n option expects an integer.\n", stderr); - return EXIT_FAILURE; - } - - return client_msg(disp, DefaultRootWindow(disp), "_NET_NUMBER_OF_DESKTOPS", - n, 0, 0, 0, 0); + if (sscanf(options.param, "%lu", &n) != 1) { + fputs("The -n option expects an integer.\n", stderr); + return EXIT_FAILURE; + } + + return client_msg(disp, DefaultRootWindow(disp), "_NET_NUMBER_OF_DESKTOPS", + n, 0, 0, 0, 0); }/*}}}*/ static int switch_desktop (Display *disp) {/*{{{*/ - int target = -1; - - target = atoi(options.param); - if (target == -1) { - fputs("Invalid desktop ID.\n", stderr); - return EXIT_FAILURE; - } - - return client_msg(disp, DefaultRootWindow(disp), "_NET_CURRENT_DESKTOP", - (unsigned long)target, 0, 0, 0, 0); + int target = -1; + + target = atoi(options.param); + if (target == -1) { + fputs("Invalid desktop ID.\n", stderr); + return EXIT_FAILURE; + } + + return client_msg(disp, DefaultRootWindow(disp), "_NET_CURRENT_DESKTOP", + (unsigned long)target, 0, 0, 0, 0); }/*}}}*/ static void window_set_title (Display *disp, Window win, /* {{{ */ - char *title, char mode) { - gchar *title_utf8; - gchar *title_local; - - if (envir_utf8) { - title_utf8 = g_strdup(title); - title_local = NULL; + char *title, char mode) { + gchar *title_utf8; + gchar *title_local; + + if (envir_utf8) { + title_utf8 = g_strdup(title); + title_local = NULL; + } + else { + if (! (title_utf8 = g_locale_to_utf8(title, -1, NULL, NULL, NULL))) { + title_utf8 = g_strdup(title); + } + title_local = g_strdup(title); + } + + if (mode == 'T' || mode == 'N') { + /* set name */ + if (title_local) { + XChangeProperty(disp, win, XA_WM_NAME, XA_STRING, 8, PropModeReplace, + title_local, strlen(title_local)); } else { - if (! (title_utf8 = g_locale_to_utf8(title, -1, NULL, NULL, NULL))) { - title_utf8 = g_strdup(title); - } - title_local = g_strdup(title); - } - - if (mode == 'T' || mode == 'N') { - /* set name */ - if (title_local) { - XChangeProperty(disp, win, XA_WM_NAME, XA_STRING, 8, PropModeReplace, - title_local, strlen(title_local)); - } - else { - XDeleteProperty(disp, win, XA_WM_NAME); - } - XChangeProperty(disp, win, XInternAtom(disp, "_NET_WM_NAME", False), - XInternAtom(disp, "UTF8_STRING", False), 8, PropModeReplace, - title_utf8, strlen(title_utf8)); + XDeleteProperty(disp, win, XA_WM_NAME); } + XChangeProperty(disp, win, XInternAtom(disp, "_NET_WM_NAME", False), + XInternAtom(disp, "UTF8_STRING", False), 8, PropModeReplace, + title_utf8, strlen(title_utf8)); + } - if (mode == 'T' || mode == 'I') { - /* set icon name */ - if (title_local) { - XChangeProperty(disp, win, XA_WM_ICON_NAME, XA_STRING, 8, PropModeReplace, - title_local, strlen(title_local)); - } - else { - XDeleteProperty(disp, win, XA_WM_ICON_NAME); - } - XChangeProperty(disp, win, XInternAtom(disp, "_NET_WM_ICON_NAME", False), - XInternAtom(disp, "UTF8_STRING", False), 8, PropModeReplace, - title_utf8, strlen(title_utf8)); + if (mode == 'T' || mode == 'I') { + /* set icon name */ + if (title_local) { + XChangeProperty(disp, win, XA_WM_ICON_NAME, XA_STRING, 8, PropModeReplace, + title_local, strlen(title_local)); } - - g_free(title_utf8); - g_free(title_local); - + else { + XDeleteProperty(disp, win, XA_WM_ICON_NAME); + } + XChangeProperty(disp, win, XInternAtom(disp, "_NET_WM_ICON_NAME", False), + XInternAtom(disp, "UTF8_STRING", False), 8, PropModeReplace, + title_utf8, strlen(title_utf8)); + } + + g_free(title_utf8); + g_free(title_local); + }/*}}}*/ static int window_to_desktop (Display *disp, Window win, int desktop) {/*{{{*/ - unsigned long *cur_desktop = NULL; - Window root = DefaultRootWindow(disp); - - if (desktop == -1) { - if (! (cur_desktop = (unsigned long *)get_property(disp, root, - XA_CARDINAL, "_NET_CURRENT_DESKTOP", NULL))) { - if (! (cur_desktop = (unsigned long *)get_property(disp, root, - XA_CARDINAL, "_WIN_WORKSPACE", NULL))) { - fputs("Cannot get current desktop properties. " - "(_NET_CURRENT_DESKTOP or _WIN_WORKSPACE property)" - "\n", stderr); - return EXIT_FAILURE; - } - } - desktop = *cur_desktop; + unsigned long *cur_desktop = NULL; + Window root = DefaultRootWindow(disp); + + if (desktop == -1) { + if (! (cur_desktop = (unsigned long *)get_property(disp, root, + XA_CARDINAL, "_NET_CURRENT_DESKTOP", NULL))) { + if (! (cur_desktop = (unsigned long *)get_property(disp, root, + XA_CARDINAL, "_WIN_WORKSPACE", NULL))) { + fputs("Cannot get current desktop properties. " + "(_NET_CURRENT_DESKTOP or _WIN_WORKSPACE property)" + "\n", stderr); + return EXIT_FAILURE; + } } - g_free(cur_desktop); + desktop = *cur_desktop; + } + g_free(cur_desktop); - return client_msg(disp, win, "_NET_WM_DESKTOP", (unsigned long)desktop, - 0, 0, 0, 0); + return client_msg(disp, win, "_NET_WM_DESKTOP", (unsigned long)desktop, + 0, 0, 0, 0); }/*}}}*/ static int activate_window (Display *disp, Window win, /* {{{ */ @@ -910,6 +910,9 @@ static int action_window (Display *disp, Window win, char mode) {/*{{{*/ window_set_title(disp, win, options.param, mode); return EXIT_SUCCESS; + case 'z': + return XLowerWindow(disp, win); + default: fprintf(stderr, "Unknown action: '%c'\n", mode); return EXIT_FAILURE; From 995febbdaa8ff3e83176679f03a53ae962c358e1 Mon Sep 17 00:00:00 2001 From: dan corson Date: Wed, 5 Nov 2008 11:38:30 -0800 Subject: [PATCH 02/14] adding -E for get-title --- main.c | 16 +++++++++++++--- 1 file changed, 13 insertions(+), 3 deletions(-) diff --git a/main.c b/main.c index a5d1be8..d0689e3 100644 --- a/main.c +++ b/main.c @@ -264,7 +264,7 @@ int main (int argc, char **argv) { /* {{{ */ } } - while ((opt = getopt(argc, argv, "FGVvhlupidmxa:r:s:c:t:w:k:o:n:g:e:b:z:N:I:T:R:")) != -1) { + while ((opt = getopt(argc, argv, "FGVvhlupidmxa:r:s:c:t:w:k:o:n:g:e:b:z:E:N:I:T:R:")) != -1) { missing_option = 0; switch (opt) { case 'F': @@ -289,7 +289,7 @@ int main (int argc, char **argv) { /* {{{ */ case 'p': options.show_pid = 1; break; - case 'a': case 'c': case 'R': case 'z': + case 'a': case 'c': case 'R': case 'z': case 'E': options.param_window = optarg; action = opt; break; @@ -355,7 +355,7 @@ int main (int argc, char **argv) { /* {{{ */ case 'm': ret = wm_info(disp); break; - case 'a': case 'c': case 'R': case 'z': + case 'a': case 'c': case 'R': case 'z': case 'E': case 't': case 'e': case 'b': case 'N': case 'I': case 'T': if (! options.param_window) { fputs("No window was specified.\n", stderr); @@ -875,6 +875,8 @@ static int window_move_resize (Display *disp, Window win, char *arg) {/*{{{*/ }/*}}}*/ static int action_window (Display *disp, Window win, char mode) {/*{{{*/ + XTextProperty text_prop_return; + p_verbose("Using window: 0x%.8lx\n", win); switch (mode) { case 'a': @@ -911,7 +913,15 @@ static int action_window (Display *disp, Window win, char mode) {/*{{{*/ return EXIT_SUCCESS; case 'z': + // iconify return XLowerWindow(disp, win); + case 'E': + // say title + // FIXME: why isn't XSetWMName used for the other title stuff in + // wmctrl? + XGetWMName(disp, win, &text_prop_return); + printf("%s\n", text_prop_return.value); + return EXIT_SUCCESS; default: fprintf(stderr, "Unknown action: '%c'\n", mode); From aec57b57f0d2694e1e0a317d874a6dc6883db430 Mon Sep 17 00:00:00 2001 From: dan corson Date: Wed, 5 Nov 2008 13:10:07 -0800 Subject: [PATCH 03/14] fix -E get-title to work with utf8/_NET_WM_NAME correctly --- main.c | 16 ++++++++-------- 1 file changed, 8 insertions(+), 8 deletions(-) diff --git a/main.c b/main.c index d0689e3..927c519 100644 --- a/main.c +++ b/main.c @@ -874,9 +874,14 @@ static int window_move_resize (Display *disp, Window win, char *arg) {/*{{{*/ } }/*}}}*/ -static int action_window (Display *disp, Window win, char mode) {/*{{{*/ - XTextProperty text_prop_return; +static int window_say_title (Display *disp, Window win) { + gchar *title_utf8 = get_window_title(disp, win); + printf("%s\n", title_utf8); + g_free(title_utf8); + return EXIT_SUCCESS; +} +static int action_window (Display *disp, Window win, char mode) {/*{{{*/ p_verbose("Using window: 0x%.8lx\n", win); switch (mode) { case 'a': @@ -916,12 +921,7 @@ static int action_window (Display *disp, Window win, char mode) {/*{{{*/ // iconify return XLowerWindow(disp, win); case 'E': - // say title - // FIXME: why isn't XSetWMName used for the other title stuff in - // wmctrl? - XGetWMName(disp, win, &text_prop_return); - printf("%s\n", text_prop_return.value); - return EXIT_SUCCESS; + return window_say_title(disp, win); default: fprintf(stderr, "Unknown action: '%c'\n", mode); From b521f37d7f376e43ad7f64342b7e36b80182c7db Mon Sep 17 00:00:00 2001 From: Chris Piro Date: Thu, 20 Nov 2008 04:54:36 -0800 Subject: [PATCH 04/14] add -r -y: just like -e but reactivate after the move --- main.c | 14 +++++++++++--- 1 file changed, 11 insertions(+), 3 deletions(-) diff --git a/main.c b/main.c index 927c519..a9b98a9 100644 --- a/main.c +++ b/main.c @@ -58,6 +58,7 @@ Free Software Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. " -r -t Move the window to the specified desktop.\n" \ " -r -e Resize and move the window around the desktop.\n" \ " The format of the argument is described below.\n" \ +" -r -y Resize and move like above, then reactivate.\n" \ " -r -b Change the state of the window. Using this option it's\n" \ " possible for example to make the window maximized,\n" \ " minimized or fullscreen. The format of the \n" \ @@ -264,7 +265,7 @@ int main (int argc, char **argv) { /* {{{ */ } } - while ((opt = getopt(argc, argv, "FGVvhlupidmxa:r:s:c:t:w:k:o:n:g:e:b:z:E:N:I:T:R:")) != -1) { + while ((opt = getopt(argc, argv, "FGVvhlupidmxa:r:s:c:t:w:k:o:n:g:e:y:b:z:E:N:I:T:R:")) != -1) { missing_option = 0; switch (opt) { case 'F': @@ -296,7 +297,7 @@ int main (int argc, char **argv) { /* {{{ */ case 'r': options.param_window = optarg; break; - case 't': case 'e': case 'b': case 'N': case 'I': case 'T': + case 't': case 'e': case 'b': case 'N': case 'I': case 'T': case 'y': options.param = optarg; action = opt; break; @@ -356,7 +357,7 @@ int main (int argc, char **argv) { /* {{{ */ ret = wm_info(disp); break; case 'a': case 'c': case 'R': case 'z': case 'E': - case 't': case 'e': case 'b': case 'N': case 'I': case 'T': + case 't': case 'e': case 'b': case 'N': case 'I': case 'T': case 'y': if (! options.param_window) { fputs("No window was specified.\n", stderr); return EXIT_FAILURE; @@ -882,6 +883,7 @@ static int window_say_title (Display *disp, Window win) { } static int action_window (Display *disp, Window win, char mode) {/*{{{*/ + int rv; p_verbose("Using window: 0x%.8lx\n", win); switch (mode) { case 'a': @@ -894,6 +896,12 @@ static int action_window (Display *disp, Window win, char mode) {/*{{{*/ /* resize/move the window around the desktop => -r -e */ return window_move_resize(disp, win, options.param); + case 'y': + /* resize/move the window, then activate it */ + rv = window_move_resize(disp, win, options.param); + activate_window(disp, win, TRUE); + return rv; + case 'b': /* change state of a window => -r -b */ return window_state(disp, win, options.param); From 0528a3410ee114a99cfe394dfd52cddda18c3d8b Mon Sep 17 00:00:00 2001 From: Kevin Der Date: Mon, 12 Jan 2009 14:51:57 -0800 Subject: [PATCH 05/14] add -j: list current desktop --- main.c | 26 +++++++++++++++++++++++++- 1 file changed, 25 insertions(+), 1 deletion(-) diff --git a/main.c b/main.c index a9b98a9..be15390 100644 --- a/main.c +++ b/main.c @@ -49,6 +49,7 @@ Free Software Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. " -l List windows managed by the window manager.\n" \ " -d List desktops. The current desktop is marked\n" \ " with an asterisk.\n" \ +" -j List current desktop.\n" \ " -s Switch to the specified desktop.\n" \ " -a Activate the window by switching to its desktop and\n" \ " raising it.\n" \ @@ -195,6 +196,7 @@ static int client_msg(Display *disp, Window win, char *msg, unsigned long data2, unsigned long data3, unsigned long data4); static int list_windows (Display *disp); +static int list_current_desktop (Display *disp); static int list_desktops (Display *disp); static int showing_desktop (Display *disp); static int change_viewport (Display *disp); @@ -265,7 +267,7 @@ int main (int argc, char **argv) { /* {{{ */ } } - while ((opt = getopt(argc, argv, "FGVvhlupidmxa:r:s:c:t:w:k:o:n:g:e:y:b:z:E:N:I:T:R:")) != -1) { + while ((opt = getopt(argc, argv, "FGVvhlupidjmxa:r:s:c:t:w:k:o:n:g:e:y:b:z:E:N:I:T:R:")) != -1) { missing_option = 0; switch (opt) { case 'F': @@ -347,6 +349,9 @@ int main (int argc, char **argv) { /* {{{ */ case 'l': ret = list_windows(disp); break; + case 'j': + ret = list_current_desktop(disp); + break; case 'd': ret = list_desktops(disp); break; @@ -1036,6 +1041,25 @@ static int action_window_str (Display *disp, char mode) {/*{{{*/ } }/*}}}*/ + +static int list_current_desktop (Display *disp) {/*{{{*/ + unsigned long *cur_desktop = NULL; + Window root = DefaultRootWindow(disp); + if (! (cur_desktop = (unsigned long *)get_property(disp, root, + XA_CARDINAL, "_NET_CURRENT_DESKTOP", NULL))) { + if (! (cur_desktop = (unsigned long *)get_property(disp, root, + XA_CARDINAL, "_WIN_WORKSPACE", NULL))) { + fputs("Cannot get current desktop properties. " + "(_NET_CURRENT_DESKTOP or _WIN_WORKSPACE property)" + "\n", stderr); + g_free(cur_desktop); + return EXIT_FAILURE; + } + } + printf("%-2d\n", *cur_desktop); + return EXIT_SUCCESS; +} + static int list_desktops (Display *disp) {/*{{{*/ unsigned long *num_desktops = NULL; unsigned long *cur_desktop = NULL; From 4b96db738a110435d1ed3ecb467c02759cf4cf62 Mon Sep 17 00:00:00 2001 From: Daniel Corson Date: Fri, 16 Jan 2009 23:55:40 -0800 Subject: [PATCH 06/14] fix for 64bit --- main.c | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/main.c b/main.c index be15390..75193eb 100644 --- a/main.c +++ b/main.c @@ -1485,7 +1485,7 @@ static gchar *get_property (Display *disp, Window win, /*{{{*/ } /* null terminate the result to make string handling easier */ - tmp_size = (ret_format / 8) * ret_nitems; + tmp_size = (ret_format / (32 / sizeof(long))) * ret_nitems; ret = g_malloc(tmp_size + 1); memcpy(ret, ret_prop, tmp_size); ret[tmp_size] = '\0'; From 4d08030463819d6479bc138af522cbdb9a222a59 Mon Sep 17 00:00:00 2001 From: Vadim Ushakov Date: Sat, 10 Nov 2012 13:00:16 +0800 Subject: [PATCH 07/14] Add option -S to sort window list in stacking order --- main.c | 40 +++++++++++++++++++++++++++------------- wmctrl.1 | 4 ++++ 2 files changed, 31 insertions(+), 13 deletions(-) diff --git a/main.c b/main.c index 75193eb..fc86894 100644 --- a/main.c +++ b/main.c @@ -81,6 +81,7 @@ Free Software Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. " -h Print help.\n" \ "\n" \ "Options:\n" \ +" -S List windows in stacking order (bottom to top).\n" \ " -i Interpret as a numerical window ID.\n" \ " -p Include PIDs in the window list. Very few\n" \ " X applications support this feature.\n" \ @@ -232,6 +233,7 @@ static struct { int show_class; int show_pid; int show_geometry; + int stacking_order; int match_by_id; int match_by_cls; int full_window_title_match; @@ -267,12 +269,15 @@ int main (int argc, char **argv) { /* {{{ */ } } - while ((opt = getopt(argc, argv, "FGVvhlupidjmxa:r:s:c:t:w:k:o:n:g:e:y:b:z:E:N:I:T:R:")) != -1) { + while ((opt = getopt(argc, argv, "FGVvhSlupidjmxa:r:s:c:t:w:k:o:n:g:e:y:b:z:E:N:I:T:R:")) != -1) { missing_option = 0; switch (opt) { case 'F': options.full_window_title_match = 1; break; + case 'S': + options.stacking_order = 1; + break; case 'G': options.show_geometry = 1; break; @@ -1304,18 +1309,27 @@ static int longest_str (gchar **strv) {/*{{{*/ }/*}}}*/ static Window *get_client_list (Display *disp, unsigned long *size) {/*{{{*/ - Window *client_list; - - if ((client_list = (Window *)get_property(disp, DefaultRootWindow(disp), - XA_WINDOW, "_NET_CLIENT_LIST", size)) == NULL) { - if ((client_list = (Window *)get_property(disp, DefaultRootWindow(disp), - XA_CARDINAL, "_WIN_CLIENT_LIST", size)) == NULL) { - fputs("Cannot get client list properties. \n" - "(_NET_CLIENT_LIST or _WIN_CLIENT_LIST)" - "\n", stderr); - return NULL; - } - } + Window *client_list = NULL; + char * msg = NULL; + + if (options.stacking_order) + { + msg = "_NET_CLIENT_LIST_STACKING"; + client_list = (Window *) get_property(disp, DefaultRootWindow(disp), + XA_WINDOW, "_NET_CLIENT_LIST_STACKING", size); + } + else + { + msg = "_NET_CLIENT_LIST or _WIN_CLIENT_LIST"; + client_list = (Window *)get_property(disp, DefaultRootWindow(disp), + XA_WINDOW, "_NET_CLIENT_LIST", size); + if (!client_list) + client_list = (Window *)get_property(disp, DefaultRootWindow(disp), + XA_CARDINAL, "_WIN_CLIENT_LIST", size); + } + + if (!client_list) + fprintf(stderr, "Cannot get client list properties.\n(%s)\n", msg); return client_list; }/*}}}*/ diff --git a/wmctrl.1 b/wmctrl.1 index 1a5dfc6..ac893c7 100644 --- a/wmctrl.1 +++ b/wmctrl.1 @@ -219,6 +219,10 @@ Include geometry information in the output of the .B \-l action. +.TP +.B \-S +List windows in stacking order (bottom to top). + .TP .B \-i Interpret window arguments From ce8aa4ebfa32da5909336c4676ecdb58ad0b389d Mon Sep 17 00:00:00 2001 From: Vadim Ushakov Date: Sat, 10 Nov 2012 14:24:45 +0800 Subject: [PATCH 08/14] Add support for _OB_WM_STATE_UNDECORATED. Allow usage of direct WM_STATE names. --- main.c | 40 ++++++++++++++++++++++++++++++++-------- 1 file changed, 32 insertions(+), 8 deletions(-) diff --git a/main.c b/main.c index fc86894..c8931b6 100644 --- a/main.c +++ b/main.c @@ -745,6 +745,33 @@ static int close_window (Display *disp, Window win) {/*{{{*/ 0, 0, 0, 0, 0); }/*}}}*/ +static gchar * normalize_wm_state_name(const char * name) +{ + char * short_names[] = { + "modal", "sticky", "maximized_vert", "maximized_horz", + "shaded", "skip_taskbar", "skip_pager", "hidden", + "fullscreen", "above", "below", 0}; + + int i; + for (i = 0; short_names[i]; i++) + { + if (strcmp(short_names[i], name) == 0) + { + gchar * upcase = g_ascii_strup(name, -1); + gchar * result = g_strdup_printf("_NET_WM_STATE_%s", upcase); + g_free(upcase); + return result; + } + } + + if (strcmp("undecorated", name) == 0) + { + return g_strdup("_OB_WM_STATE_UNDECORATED"); + } + + return g_strdup(name); +} + static int window_state (Display *disp, Window win, char *arg) {/*{{{*/ unsigned long action; Atom prop1 = 0; @@ -758,8 +785,8 @@ static int window_state (Display *disp, Window win, char *arg) {/*{{{*/ } if ((p1 = strchr(arg, ','))) { - gchar *tmp_prop1, *tmp1; - + gchar *tmp_prop1; + *p1 = '\0'; /* action */ @@ -780,17 +807,16 @@ static int window_state (Display *disp, Window win, char *arg) {/*{{{*/ /* the second property */ if ((p2 = strchr(p1, ','))) { - gchar *tmp_prop2, *tmp2; + gchar *tmp_prop2; *p2 = '\0'; p2++; if (strlen(p2) == 0) { fputs("Invalid zero length property.\n", stderr); return EXIT_FAILURE; } - tmp_prop2 = g_strdup_printf("_NET_WM_STATE_%s", tmp2 = g_ascii_strup(p2, -1)); + tmp_prop2 = normalize_wm_state_name(p2); p_verbose("State 2: %s\n", tmp_prop2); prop2 = XInternAtom(disp, tmp_prop2, False); - g_free(tmp2); g_free(tmp_prop2); } @@ -799,13 +825,11 @@ static int window_state (Display *disp, Window win, char *arg) {/*{{{*/ fputs("Invalid zero length property.\n", stderr); return EXIT_FAILURE; } - tmp_prop1 = g_strdup_printf("_NET_WM_STATE_%s", tmp1 = g_ascii_strup(p1, -1)); + tmp_prop1 = normalize_wm_state_name(p1); p_verbose("State 1: %s\n", tmp_prop1); prop1 = XInternAtom(disp, tmp_prop1, False); - g_free(tmp1); g_free(tmp_prop1); - return client_msg(disp, win, "_NET_WM_STATE", action, (unsigned long)prop1, (unsigned long)prop2, 0, 0); } From 894bb1db230b55f1b355362f52940fc7141546d5 Mon Sep 17 00:00:00 2001 From: Vadim Ushakov Date: Sat, 10 Nov 2012 21:17:50 +0800 Subject: [PATCH 09/14] Add action -Y (iconify). --- main.c | 19 ++++++++++++++----- wmctrl.1 | 6 ++++++ 2 files changed, 20 insertions(+), 5 deletions(-) diff --git a/main.c b/main.c index c8931b6..53fa4a2 100644 --- a/main.c +++ b/main.c @@ -56,13 +56,14 @@ Free Software Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. " -c Close the window gracefully.\n" \ " -R Move the window to the current desktop and\n" \ " activate it.\n" \ +" -Y Iconify (minimize) the window.\n" \ " -r -t Move the window to the specified desktop.\n" \ " -r -e Resize and move the window around the desktop.\n" \ " The format of the argument is described below.\n" \ " -r -y Resize and move like above, then reactivate.\n" \ " -r -b Change the state of the window. Using this option it's\n" \ " possible for example to make the window maximized,\n" \ -" minimized or fullscreen. The format of the \n" \ +" shaded or fullscreen. The format of the \n" \ " argument and list of possible states is given below.\n" \ " -r -N Set the name (long title) of the window.\n" \ " -r -I Set the icon name (short title) of the window.\n" \ @@ -252,10 +253,10 @@ int main (int argc, char **argv) { /* {{{ */ Display *disp; memset(&options, 0, sizeof(options)); /* just for sure */ - + /* necessary to make g_get_charset() and g_locale_*() work */ setlocale(LC_ALL, ""); - + /* make "--help" and "--version" work. I don't want to use * getopt_long for portability reasons */ if (argc == 2 && argv[1]) { @@ -269,7 +270,7 @@ int main (int argc, char **argv) { /* {{{ */ } } - while ((opt = getopt(argc, argv, "FGVvhSlupidjmxa:r:s:c:t:w:k:o:n:g:e:y:b:z:E:N:I:T:R:")) != -1) { + while ((opt = getopt(argc, argv, "FGVvhSlupidjmxa:r:s:c:t:w:k:o:n:g:e:y:b:z:E:N:I:T:R:Y:")) != -1) { missing_option = 0; switch (opt) { case 'F': @@ -297,7 +298,7 @@ int main (int argc, char **argv) { /* {{{ */ case 'p': options.show_pid = 1; break; - case 'a': case 'c': case 'R': case 'z': case 'E': + case 'a': case 'c': case 'R': case 'z': case 'Y': case 'E': options.param_window = optarg; action = opt; break; @@ -368,6 +369,7 @@ int main (int argc, char **argv) { /* {{{ */ break; case 'a': case 'c': case 'R': case 'z': case 'E': case 't': case 'e': case 'b': case 'N': case 'I': case 'T': case 'y': + case 'Y': if (! options.param_window) { fputs("No window was specified.\n", stderr); return EXIT_FAILURE; @@ -740,6 +742,10 @@ static int activate_window (Display *disp, Window win, /* {{{ */ return EXIT_SUCCESS; }/*}}}*/ +static int iconify_window (Display *disp, Window win) {/* {{{ */ + return !XIconifyWindow(disp, win, DefaultScreen(disp)); +}/*}}}*/ + static int close_window (Display *disp, Window win) {/*{{{*/ return client_msg(disp, win, "_NET_CLOSE_WINDOW", 0, 0, 0, 0, 0); @@ -923,6 +929,9 @@ static int action_window (Display *disp, Window win, char mode) {/*{{{*/ case 'a': return activate_window(disp, win, TRUE); + case 'Y': + return iconify_window(disp, win); + case 'c': return close_window(disp, win); diff --git a/wmctrl.1 b/wmctrl.1 index ac893c7..59d57dd 100644 --- a/wmctrl.1 +++ b/wmctrl.1 @@ -71,6 +71,12 @@ Close the window .I gracefully. +.TP +.BI \-Y " " +Iconify the window +.I +\[char46] + .TP .B \-d List all desktops managed by the window manager. One line is output From d68cf082030e00ba8f4d63fe8a096d26da6acdaa Mon Sep 17 00:00:00 2001 From: r2rien Date: Sun, 10 Feb 2013 11:58:20 +0100 Subject: [PATCH 10/14] 64-bit-data backported from debian patch [Conflict resolution note from kfogel: When I cherry-picked this commit, I resolved the conflict it had with the slightly different approach taken in earlier cherry-picked commit 4b96db738a. I chose to just use this commit, because the conditional check of ret_format's value seemed like a careful way to do things. This meant manually reverting the change from commit 4b96db738a; however, in practice I believe either way would work on any 32-bit or 64-bit architecture.] --- main.c | 15 ++++++++++++++- 1 file changed, 14 insertions(+), 1 deletion(-) diff --git a/main.c b/main.c index 53fa4a2..39fd467 100644 --- a/main.c +++ b/main.c @@ -1517,6 +1517,16 @@ static gchar *get_property (Display *disp, Window win, /*{{{*/ * * long_length = Specifies the length in 32-bit multiples of the * data to be retrieved. + * + * NOTE: see + * http://mail.gnome.org/archives/wm-spec-list/2003-March/msg00067.html + * In particular: + * + * When the X window system was ported to 64-bit architectures, a + * rather peculiar design decision was made. 32-bit quantities such + * as Window IDs, atoms, etc, were kept as longs in the client side + * APIs, even when long was changed to 64 bits. + * */ if (XGetWindowProperty(disp, win, xa_prop_name, 0, MAX_PROPERTY_VALUE_LEN / 4, False, xa_prop_type, &xa_ret_type, &ret_format, @@ -1532,7 +1542,10 @@ static gchar *get_property (Display *disp, Window win, /*{{{*/ } /* null terminate the result to make string handling easier */ - tmp_size = (ret_format / (32 / sizeof(long))) * ret_nitems; + + tmp_size = (ret_format / 8) * ret_nitems; + /* Correct 64 Architecture implementation of 32 bit data */ + if(ret_format==32) tmp_size *= sizeof(long)/4; ret = g_malloc(tmp_size + 1); memcpy(ret, ret_prop, tmp_size); ret[tmp_size] = '\0'; From f2ba0c06293e3cbe909e0d950c12c3ad247e34d8 Mon Sep 17 00:00:00 2001 From: Karl Fogel Date: Sun, 9 Sep 2018 17:06:47 -0500 Subject: [PATCH 11/14] Cast to avoid compiler warnings --- main.c | 10 +++++----- 1 file changed, 5 insertions(+), 5 deletions(-) diff --git a/main.c b/main.c index 39fd467..453f7c5 100644 --- a/main.c +++ b/main.c @@ -661,28 +661,28 @@ static void window_set_title (Display *disp, Window win, /* {{{ */ /* set name */ if (title_local) { XChangeProperty(disp, win, XA_WM_NAME, XA_STRING, 8, PropModeReplace, - title_local, strlen(title_local)); + (guchar *) title_local, strlen(title_local)); } else { XDeleteProperty(disp, win, XA_WM_NAME); } XChangeProperty(disp, win, XInternAtom(disp, "_NET_WM_NAME", False), XInternAtom(disp, "UTF8_STRING", False), 8, PropModeReplace, - title_utf8, strlen(title_utf8)); + (guchar *) title_utf8, strlen(title_utf8)); } if (mode == 'T' || mode == 'I') { /* set icon name */ if (title_local) { XChangeProperty(disp, win, XA_WM_ICON_NAME, XA_STRING, 8, PropModeReplace, - title_local, strlen(title_local)); + (guchar *) title_local, strlen(title_local)); } else { XDeleteProperty(disp, win, XA_WM_ICON_NAME); } XChangeProperty(disp, win, XInternAtom(disp, "_NET_WM_ICON_NAME", False), XInternAtom(disp, "UTF8_STRING", False), 8, PropModeReplace, - title_utf8, strlen(title_utf8)); + (guchar *) title_utf8, strlen(title_utf8)); } g_free(title_utf8); @@ -1094,7 +1094,7 @@ static int list_current_desktop (Display *disp) {/*{{{*/ return EXIT_FAILURE; } } - printf("%-2d\n", *cur_desktop); + printf("%-2d\n", *((int *) cur_desktop)); return EXIT_SUCCESS; } From c213ccab24479aea0bce1dc2f4f0abb76c7338d6 Mon Sep 17 00:00:00 2001 From: Matthew Austin Date: Tue, 4 Apr 2017 12:05:04 -0700 Subject: [PATCH 12/14] Fix typo in help printout "Th first matching" -> "The first matching" --- main.c | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/main.c b/main.c index 453f7c5..af4ad00 100644 --- a/main.c +++ b/main.c @@ -113,7 +113,7 @@ Free Software Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. "\n" \ " The -x option may be used to interpret the argument\n" \ " as a string, which is matched against the window's\n" \ -" class name (WM_CLASS property). Th first matching\n" \ +" class name (WM_CLASS property). The first matching\n" \ " window is used. The matching isn't case sensitive\n" \ " and the string may appear in any position\n" \ " of the class name. So it's recommended to always use\n" \ From 0833092966e16144c76186b2c9ca4ebcdc709111 Mon Sep 17 00:00:00 2001 From: Karl Fogel Date: Sun, 9 Sep 2018 17:48:26 -0500 Subject: [PATCH 13/14] Update README to reflect new situation --- README | 48 ++++++++++++++++++++++++++++-------------------- 1 file changed, 28 insertions(+), 20 deletions(-) diff --git a/README b/README index 4622381..0ea3601 100644 --- a/README +++ b/README @@ -1,21 +1,33 @@ ############################################################################## # # -# This is the Conservatory archive of the wmctrl, a Unix command-line # -# tool to interact with an EWMH/NetWM-compatible X Window Manager. # +# wmctrl is free software (open source software), a Unix command-line # +# tool for interacting with an EWMH/NetWM-compatible X Window Manager. # # # -# wmctrl is free software (open source software), written by Tomas Styblo # -# with contributions from others. http://tripie.sweb.cz/utils/wmctrl/ is # -# its home page. # +# wmctrl was originally written by Tomas Styblo with contributions from # +# from others. Its home page is http://tripie.sweb.cz/utils/wmctrl/. # +# # +# The version here is enhanced with various changes that people around # +# the Net had added in their own divergent versions. This version is # +# meant to gather all of those improvements together in one place. # +# It also contains the full release history of wmctrl (one commit # +# per release) because it starts from the Conservatory archive at # +# https://github.com/Conservatory/wmctrl. The relevant part of the # +# Conservatory's README is below, lightly edited. # +# # +# ===================================================================== # # # # Because wmctrl does not appear to have any publicly accessible version # -# control repository, we've created this Conservatory Git repository. # -# We're not sure if Tomas Styblo or any of the other contributors are # -# are currently accepting bug reports, but if you run into a bug you # -# could try reporting it upstream and see what happens. (As of this # -# writing on 2018-09-08, the latest wmctrl release was in January 2005.) # +# control repository, we created a Conservatory Git repository for it. # +# We don't know if Tomas Styblo still accepts bug reports for wmctrl, # +# but if you run into a bug you could try reporting it to him and see # +# what happens. (As of this writing on 2018-09-09, the latest official # +# wmctrl release was 1.07 in January 2005.) Please also feel free to # +# file bugs here. Although I don't consider this copy of wmctrl to be # +# "official" and I'm certainly not committing to maintain it, I'll merge # +# easy fixes or enhancements if they don't take too much time to review. # # # -# About the Conservatory and its copy of wmctrl: # -# ---------------------------------------------- # +# About the Conservatory copy of wmctrl that this started from: # +# ------------------------------------------------------------- # # # # The Conservatory (https://conservatory.github.io/) is a place where # # free software that has no other public version-controlled home on the # @@ -39,14 +51,10 @@ # Adds -j (list current desktop), -z (lower window), # # and some other interesting new options. # # # -# Those two started from a top-skim of wmctrl 1.07, though, so they don't # -# preserve the full upstream release history. It'd be nice if someone # -# were to clone this Conservatory copy and then re-port the enhancements # -# from those other repositories to create a canonical new wmctrl release. # -# # -# In the meantime, this README is the only file we've changed from what's # -# in upstream, and the only change we made was to add this note. Below # -# is the original content of the README file. # +# Those two started from a top-skim of wmctrl 1.07, though, so they # +# didn't preserve the full upstream release history. Later @kfogel # +# cloned the Conservatory copy and then merged those enhancements into # +# his copy. Below is the original content of the upstream README file. # # # ############################################################################## From c408c9e1c00092fc9aea2bd1ba4bec8e9ef3c40c Mon Sep 17 00:00:00 2001 From: Karl Fogel Date: Mon, 10 Sep 2018 11:33:00 -0500 Subject: [PATCH 14/14] Explain maintenance situation even more clearly --- README | 66 ++++++++++++++++++++-------------------------------------- 1 file changed, 22 insertions(+), 44 deletions(-) diff --git a/README b/README index 0ea3601..6a41423 100644 --- a/README +++ b/README @@ -6,55 +6,33 @@ # wmctrl was originally written by Tomas Styblo with contributions from # # from others. Its home page is http://tripie.sweb.cz/utils/wmctrl/. # # # -# The version here is enhanced with various changes that people around # -# the Net had added in their own divergent versions. This version is # -# meant to gather all of those improvements together in one place. # -# It also contains the full release history of wmctrl (one commit # -# per release) because it starts from the Conservatory archive at # -# https://github.com/Conservatory/wmctrl. The relevant part of the # -# Conservatory's README is below, lightly edited. # +# The version here contains various enhancements that people around the # +# Net had added in their own divergent versions. The two main sources # +# of these enhancements were https://github.com/dancor/wmctrl and # +# https://github.com/geekless/wmctrl. The additions include these # +# new command-line options/actions: # # # -# ===================================================================== # +# -Y: iconify (Vadim Ushakov) # +# -S: sort window list in stacking order (Vadim Ushakov) # +# -j: list current desktop (Kevin Der) # +# -r -y: like -e but reactivate after the move (Chris Piro ) # +# -E: get-title (Dan Corson) # +# -z: lower window (Dan Corson) # # # -# Because wmctrl does not appear to have any publicly accessible version # -# control repository, we created a Conservatory Git repository for it. # -# We don't know if Tomas Styblo still accepts bug reports for wmctrl, # -# but if you run into a bug you could try reporting it to him and see # -# what happens. (As of this writing on 2018-09-09, the latest official # -# wmctrl release was 1.07 in January 2005.) Please also feel free to # -# file bugs here. Although I don't consider this copy of wmctrl to be # -# "official" and I'm certainly not committing to maintain it, I'll merge # -# easy fixes or enhancements if they don't take too much time to review. # +# Other improvements are a typo fix for the --help output, and some # +# fixes for build-time compiler warnings. # # # -# About the Conservatory copy of wmctrl that this started from: # -# ------------------------------------------------------------- # +# This repository also contains the full release history of wmctrl (one # +# commit per release), because it is based on the Conservatory archive # +# at https://github.com/Conservatory/wmctrl. # # # -# The Conservatory (https://conservatory.github.io/) is a place where # -# free software that has no other public version-controlled home on the # -# Internet is made available in a git repository. For wmctrl, we started # -# from http://tripie.sweb.cz/utils/wmctrl/dist/wmctrl-1.07.tar.gz and # -# then deduced the names of the previous releases. We have committed # -# each upstream release as one git revision, starting from 1.00, using # -# https://github.com/Conservatory/conservatory.github.io/blob/master/\ # -# conservatory-import and the upstream ChangeLog to automate the process. # +# I don't know if Tomas Styblo still accepts bug reports for wmctrl, as # +# the last version he released was 1.07 in January 2005. Please feel to # +# file bug reports and enhancements here. Although I don't consider this # +# copy to be "official", and can't commit to putting in much effort as # +# a maintainer, I'll try to review and merge easy fixes and enhancements. # # # -# There are other version-controlled copies of wmctrl on the Internet. # -# We found at least these two, and there may be others: # -# # -# * https://github.com/geekless/wmctrl # -# # -# Adds -Y (iconify) and -S (sort in stacking order) options. # -# Also backports some 64-bit-data fixes from a Debian patch. # -# # -# * https://github.com/dancor/wmctrl # -# # -# Adds -j (list current desktop), -z (lower window), # -# and some other interesting new options. # -# # -# Those two started from a top-skim of wmctrl 1.07, though, so they # -# didn't preserve the full upstream release history. Later @kfogel # -# cloned the Conservatory copy and then merged those enhancements into # -# his copy. Below is the original content of the upstream README file. # +# Below is the original content of the upstream 1.07 README. # # # ##############################################################################