diff options
-rw-r--r-- | src/xserver-launcher.c | 211 |
1 files changed, 124 insertions, 87 deletions
diff --git a/src/xserver-launcher.c b/src/xserver-launcher.c index ed2fd6f..3789281 100644 --- a/src/xserver-launcher.c +++ b/src/xserver-launcher.c @@ -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; |