xwm: Update window title when window property change
authorKristian Høgsberg <krh@bitplanet.net>
Tue, 15 May 2012 14:12:57 +0000 (10:12 -0400)
committerKristian Høgsberg <krh@bitplanet.net>
Tue, 15 May 2012 14:55:56 +0000 (10:55 -0400)
src/xserver-launcher.c

index ed2fd6f..3789281 100644 (file)
@@ -118,11 +118,14 @@ struct weston_wm {
 };
 
 struct weston_wm_window {
+       struct weston_wm *wm;
        xcb_window_t id;
        xcb_window_t frame_id;
        struct weston_surface *surface;
        struct shell_surface *shsurf;
        struct wl_listener surface_destroy_listener;
+       int repaint_scheduled;
+       int properties_dirty;
        char *class;
        char *name;
        struct weston_wm_window *transient_for;
@@ -246,6 +249,87 @@ dump_window_properties(struct weston_wm *wm, xcb_window_t window)
        free(list_reply);
 }
 
+/* We reuse some predefined, but otherwise useles atoms */
+#define TYPE_WM_PROTOCOLS XCB_ATOM_CUT_BUFFER0
+
+static void
+weston_wm_window_read_properties(struct weston_wm_window *window)
+{
+       struct weston_wm *wm = window->wm;
+
+#define F(field) offsetof(struct weston_wm_window, field)
+       const struct {
+               xcb_atom_t atom;
+               xcb_atom_t type;
+               int offset;
+       } props[] = {
+               { XCB_ATOM_WM_CLASS, XCB_ATOM_STRING, F(class) },
+               { XCB_ATOM_WM_TRANSIENT_FOR, XCB_ATOM_WINDOW, F(transient_for) },
+               { wm->atom.wm_protocols, TYPE_WM_PROTOCOLS, F(protocols) },
+               { wm->atom.net_wm_window_type, XCB_ATOM_ATOM, F(type) },
+               { wm->atom.net_wm_name, XCB_ATOM_STRING, F(name) },
+       };
+#undef F
+
+       xcb_get_property_cookie_t cookie[ARRAY_LENGTH(props)];
+       xcb_get_property_reply_t *reply;
+       void *p;
+       uint32_t *xid;
+       xcb_atom_t *atom;
+       uint32_t i;
+
+       if (!window->properties_dirty)
+               return;
+       window->properties_dirty = 0;
+
+       dump_window_properties(wm, window->id);
+
+       for (i = 0; i < ARRAY_LENGTH(props); i++)
+               cookie[i] = xcb_get_property(wm->conn,
+                                            0, /* delete */
+                                            window->id,
+                                            props[i].atom,
+                                            XCB_ATOM_ANY, 0, 2048);
+
+       for (i = 0; i < ARRAY_LENGTH(props); i++)  {
+               reply = xcb_get_property_reply(wm->conn, cookie[i], NULL);
+               if (!reply)
+                       /* Bad window, typically */
+                       continue;
+               if (reply->type == XCB_ATOM_NONE) {
+                       /* No such property */
+                       free(reply);
+                       continue;
+               }
+
+               p = ((char *) window + props[i].offset);
+
+               switch (props[i].type) {
+               case XCB_ATOM_STRING:
+                       /* FIXME: We're using this for both string and
+                          utf8_string */
+                       *(char **) p =
+                               strndup(xcb_get_property_value(reply),
+                                       xcb_get_property_value_length(reply));
+                       break;
+               case XCB_ATOM_WINDOW:
+                       xid = xcb_get_property_value(reply);
+                       *(struct weston_wm_window **) p =
+                               hash_table_lookup(wm->window_hash, *xid);
+                       break;
+               case XCB_ATOM_ATOM:
+                       atom = xcb_get_property_value(reply);
+                       *(xcb_atom_t *) p = *atom;
+                       break;
+               case TYPE_WM_PROTOCOLS:
+                       break;
+               default:
+                       break;
+               }
+               free(reply);
+       }
+}
+
 static void
 data_offer_accept(struct wl_client *client, struct wl_resource *resource,
                  uint32_t time, const char *mime_type)
@@ -702,23 +786,21 @@ find_depth (xcb_connection_t *connection, int depth)
 }
 
 static void
-weston_wm_handle_expose(struct weston_wm *wm, xcb_generic_event_t *event)
+weston_wm_window_draw_decoration(void *data)
 {
+       struct weston_wm_window *window = data;
+       struct weston_wm *wm = window->wm;
        cairo_surface_t *surface;
        cairo_t *cr;
        cairo_text_extents_t text_extents;
        cairo_font_extents_t font_extents;
        xcb_render_pictforminfo_t *render_format;
-       struct weston_wm_window *window;
-       xcb_expose_event_t *expose = (xcb_expose_event_t *) event;
        int x, y, width, height;
        const char *title;
 
-       window = hash_table_lookup(wm->window_hash, expose->window);
-
-       fprintf(stderr, "XCB_EXPOSE (window %d, title %s)\n",
-               window->id, window->name);
+       weston_wm_window_read_properties(window);
 
+       window->repaint_scheduled = 0;
        width = window->width + wm->border_width * 2;
        height = window->height + wm->border_width * 2 + wm->title_height;
 
@@ -759,6 +841,32 @@ weston_wm_handle_expose(struct weston_wm *wm, xcb_generic_event_t *event)
        cairo_surface_destroy(surface);
 }
 
+static void
+weston_wm_window_schedule_repaint(struct weston_wm_window *window)
+{
+       struct weston_wm *wm = window->wm;
+
+       if (window->repaint_scheduled)
+               return;
+
+       wl_event_loop_add_idle(wm->server->loop,
+                              weston_wm_window_draw_decoration, window);
+       window->repaint_scheduled = 1;
+}
+
+static void
+weston_wm_handle_expose(struct weston_wm *wm, xcb_generic_event_t *event)
+{
+       struct weston_wm_window *window;
+       xcb_expose_event_t *expose = (xcb_expose_event_t *) event;
+
+       window = hash_table_lookup(wm->window_hash, expose->window);
+       fprintf(stderr, "XCB_EXPOSE (window %d, title %s, surface %p)\n",
+               window->id, window->name, window->surface);
+
+       weston_wm_window_schedule_repaint(window);
+}
+
 static const size_t incr_chunk_size = 64 * 1024;
 
 static void
@@ -1023,6 +1131,11 @@ weston_wm_handle_property_notify(struct weston_wm *wm, xcb_generic_event_t *even
 {
        xcb_property_notify_event_t *property_notify =
                (xcb_property_notify_event_t *) event;
+       struct weston_wm_window *window;
+
+       window = hash_table_lookup(wm->window_hash, property_notify->window);
+       if (window)
+               window->properties_dirty = 1;
 
        if (property_notify->window == wm->selection_window) {
                if (property_notify->state == XCB_PROPERTY_NEW_VALUE &&
@@ -1042,12 +1155,14 @@ weston_wm_handle_property_notify(struct weston_wm *wm, xcb_generic_event_t *even
                fprintf(stderr, "wm_protocols changed\n");
        } else if (property_notify->atom == wm->atom.net_wm_name) {
                fprintf(stderr, "_net_wm_name changed\n");
+               weston_wm_window_schedule_repaint(window);
        } else if (property_notify->atom == wm->atom.net_wm_user_time) {
                fprintf(stderr, "_net_wm_user_time changed\n");
        } else if (property_notify->atom == wm->atom.net_wm_icon_name) {
                fprintf(stderr, "_net_wm_icon_name changed\n");
        } else if (property_notify->atom == XCB_ATOM_WM_NAME) {
                fprintf(stderr, "wm_name changed\n");
+               weston_wm_window_schedule_repaint(window);
        } else if (property_notify->atom == XCB_ATOM_WM_ICON_NAME) {
                fprintf(stderr, "wm_icon_name changed\n");
        } else {
@@ -1085,6 +1200,7 @@ weston_wm_handle_create_notify(struct weston_wm *wm, xcb_generic_event_t *event)
                                     XCB_CW_EVENT_MASK, values);
 
        memset(window, 0, sizeof *window);
+       window->wm = wm;
        window->id = create_notify->window;
 
        window->width = create_notify->width;
@@ -1616,85 +1732,6 @@ get_wm_window(struct weston_surface *surface)
        return NULL;
 }
 
-/* We reuse some predefined, but otherwise useles atoms */
-#define TYPE_WM_PROTOCOLS XCB_ATOM_CUT_BUFFER0
-
-static void
-read_window_properties(struct weston_wm *wm, struct weston_wm_window *window)
-{
-#define F(field) offsetof(struct weston_wm_window, field)
-       const struct {
-               xcb_atom_t atom;
-               xcb_atom_t type;
-               int offset;
-       } props[] = {
-               { XCB_ATOM_WM_CLASS, XCB_ATOM_STRING, F(class) },
-               { XCB_ATOM_WM_TRANSIENT_FOR, XCB_ATOM_WINDOW, F(transient_for) },
-               { wm->atom.wm_protocols, TYPE_WM_PROTOCOLS, F(protocols) },
-               { wm->atom.net_wm_window_type, XCB_ATOM_ATOM, F(type) },
-               { wm->atom.net_wm_name, XCB_ATOM_STRING, F(name) },
-       };
-#undef F
-
-       xcb_get_property_cookie_t cookie[ARRAY_LENGTH(props)];
-       xcb_get_property_reply_t *reply;
-       void *p;
-       uint32_t *xid;
-       xcb_atom_t *atom;
-       uint32_t i;
-
-       dump_window_properties(wm, window->id);
-
-       for (i = 0; i < ARRAY_LENGTH(props); i++)
-               cookie[i] = xcb_get_property(wm->conn,
-                                            0, /* delete */
-                                            window->id,
-                                            props[i].atom,
-                                            XCB_ATOM_ANY, 0, 2048);
-
-       for (i = 0; i < ARRAY_LENGTH(props); i++)  {
-               reply = xcb_get_property_reply(wm->conn, cookie[i], NULL);
-               if (!reply)
-                       /* Bad window, typically */
-                       continue;
-               if (reply->type == XCB_ATOM_NONE) {
-                       /* No such property */
-                       free(reply);
-                       continue;
-               }
-
-               p = ((char *) window + props[i].offset);
-
-               switch (props[i].type) {
-               case XCB_ATOM_STRING:
-                       /* FIXME: We're using this for both string and
-                          utf8_string */
-                       *(char **) p =
-                               strndup(xcb_get_property_value(reply),
-                                       xcb_get_property_value_length(reply));
-                       break;
-               case XCB_ATOM_WINDOW:
-                       xid = xcb_get_property_value(reply);
-                       *(struct weston_wm_window **) p =
-                               hash_table_lookup(wm->window_hash, *xid);
-                       break;
-               case XCB_ATOM_ATOM:
-                       atom = xcb_get_property_value(reply);
-                       *(xcb_atom_t *) p = *atom;
-                       break;
-               case TYPE_WM_PROTOCOLS:
-                       break;
-               default:
-                       break;
-               }
-               free(reply);
-       }
-
-       fprintf(stderr, "window %d: name %s, class %s, transient_for %d\n",
-               window->id, window->name, window->class,
-               window->transient_for ? window->transient_for->id : 0);
-}
-
 static void
 xserver_set_window_id(struct wl_client *client, struct wl_resource *resource,
                      struct wl_resource *surface_resource, uint32_t id)
@@ -1717,7 +1754,7 @@ xserver_set_window_id(struct wl_client *client, struct wl_resource *resource,
 
        fprintf(stderr, "set_window_id %d for surface %p\n", id, surface);
 
-       read_window_properties(wm, window);
+       weston_wm_window_read_properties(window);
 
        window->surface = (struct weston_surface *) surface;
        window->surface_destroy_listener.notify = surface_destroy;