#define _POSIX_C_SOURCE 200809L #include <assert.h> #include <fcntl.h> #include <stdlib.h> #include <string.h> #include <unistd.h> #include <wlr/types/wlr_data_device.h> #include <wlr/util/log.h> #include <xcb/xfixes.h> #include "xwayland/selection.h" #include "xwayland/xwm.h" void xwm_selection_transfer_remove_event_source( struct wlr_xwm_selection_transfer *transfer) { if (transfer->event_source != NULL) { wl_event_source_remove(transfer->event_source); transfer->event_source = NULL; } } void xwm_selection_transfer_close_wl_client_fd( struct wlr_xwm_selection_transfer *transfer) { if (transfer->wl_client_fd >= 0) { close(transfer->wl_client_fd); transfer->wl_client_fd = -1; } } void xwm_selection_transfer_destroy_property_reply( struct wlr_xwm_selection_transfer *transfer) { free(transfer->property_reply); transfer->property_reply = NULL; } void xwm_selection_transfer_init(struct wlr_xwm_selection_transfer *transfer, struct wlr_xwm_selection *selection) { *transfer = (struct wlr_xwm_selection_transfer){ .selection = selection, .wl_client_fd = -1, }; } void xwm_selection_transfer_destroy( struct wlr_xwm_selection_transfer *transfer) { if (!transfer) { return; } xwm_selection_transfer_destroy_property_reply(transfer); xwm_selection_transfer_remove_event_source(transfer); xwm_selection_transfer_close_wl_client_fd(transfer); if (transfer->incoming_window) { struct wlr_xwm *xwm = transfer->selection->xwm; xcb_destroy_window(xwm->xcb_conn, transfer->incoming_window); xcb_flush(xwm->xcb_conn); } wl_list_remove(&transfer->link); free(transfer); } xcb_atom_t xwm_mime_type_to_atom(struct wlr_xwm *xwm, char *mime_type) { if (strcmp(mime_type, "text/plain;charset=utf-8") == 0) { return xwm->atoms[UTF8_STRING]; } else if (strcmp(mime_type, "text/plain") == 0) { return xwm->atoms[TEXT]; } xcb_intern_atom_cookie_t cookie = xcb_intern_atom(xwm->xcb_conn, 0, strlen(mime_type), mime_type); xcb_intern_atom_reply_t *reply = xcb_intern_atom_reply(xwm->xcb_conn, cookie, NULL); if (reply == NULL) { return XCB_ATOM_NONE; } xcb_atom_t atom = reply->atom; free(reply); return atom; } char *xwm_mime_type_from_atom(struct wlr_xwm *xwm, xcb_atom_t atom) { if (atom == xwm->atoms[UTF8_STRING]) { return strdup("text/plain;charset=utf-8"); } else if (atom == xwm->atoms[TEXT]) { return strdup("text/plain"); } else { return xwm_get_atom_name(xwm, atom); } } struct wlr_xwm_selection *xwm_get_selection(struct wlr_xwm *xwm, xcb_atom_t selection_atom) { if (selection_atom == xwm->atoms[CLIPBOARD]) { return &xwm->clipboard_selection; } else if (selection_atom == xwm->atoms[PRIMARY]) { return &xwm->primary_selection; } else if (selection_atom == xwm->atoms[DND_SELECTION]) { return &xwm->dnd_selection; } else { return NULL; } } static int xwm_handle_selection_property_notify(struct wlr_xwm *xwm, xcb_property_notify_event_t *event) { struct wlr_xwm_selection *selections[] = { &xwm->clipboard_selection, &xwm->primary_selection, &xwm->dnd_selection, }; for (size_t i = 0; i < sizeof(selections)/sizeof(selections[0]); ++i) { struct wlr_xwm_selection *selection = selections[i]; if (event->state == XCB_PROPERTY_NEW_VALUE && event->atom == xwm->atoms[WL_SELECTION]) { struct wlr_xwm_selection_transfer *transfer = xwm_selection_find_incoming_transfer_by_window(selection, event->window); if (transfer) { if (transfer->incr) { xwm_get_incr_chunk(transfer); } return 1; } } struct wlr_xwm_selection_transfer *outgoing; wl_list_for_each(outgoing, &selection->outgoing, link) { if (event->window == outgoing->request.requestor) { if (event->state == XCB_PROPERTY_DELETE && event->atom == outgoing->request.property && outgoing->incr) { xwm_send_incr_chunk(outgoing); } return 1; } } } return 0; } int xwm_handle_selection_event(struct wlr_xwm *xwm, xcb_generic_event_t *event) { if (xwm->seat == NULL) { wlr_log(WLR_DEBUG, "not handling selection events: " "no seat assigned to xwayland"); return 0; } switch (event->response_type & XCB_EVENT_RESPONSE_TYPE_MASK) { case XCB_SELECTION_NOTIFY: xwm_handle_selection_notify(xwm, (xcb_selection_notify_event_t *)event); return 1; case XCB_PROPERTY_NOTIFY: return xwm_handle_selection_property_notify(xwm, (xcb_property_notify_event_t *)event); case XCB_SELECTION_REQUEST: xwm_handle_selection_request(xwm, (xcb_selection_request_event_t *)event); return 1; } switch (event->response_type - xwm->xfixes->first_event) { case XCB_XFIXES_SELECTION_NOTIFY: // an X11 window has copied something to the clipboard return xwm_handle_xfixes_selection_notify(xwm, (xcb_xfixes_selection_notify_event_t *)event); } return 0; } void xwm_selection_init(struct wlr_xwm_selection *selection, struct wlr_xwm *xwm, xcb_atom_t atom) { *selection = (struct wlr_xwm_selection){ .xwm = xwm, .atom = atom, .window = xcb_generate_id(xwm->xcb_conn), }; wl_list_init(&selection->incoming); wl_list_init(&selection->outgoing); if (atom == xwm->atoms[DND_SELECTION]) { xcb_create_window( xwm->xcb_conn, XCB_COPY_FROM_PARENT, selection->window, xwm->screen->root, 0, 0, 8192, 8192, 0, XCB_WINDOW_CLASS_INPUT_ONLY, xwm->screen->root_visual, XCB_CW_EVENT_MASK, (uint32_t[]){ XCB_EVENT_MASK_SUBSTRUCTURE_NOTIFY | XCB_EVENT_MASK_PROPERTY_CHANGE } ); xcb_change_property( xwm->xcb_conn, XCB_PROP_MODE_REPLACE, selection->window, xwm->atoms[DND_AWARE], XCB_ATOM_ATOM, 32, // format 1, &(uint32_t){XDND_VERSION} ); } else { xcb_create_window( xwm->xcb_conn, XCB_COPY_FROM_PARENT, selection->window, xwm->screen->root, 0, 0, 10, 10, 0, XCB_WINDOW_CLASS_INPUT_OUTPUT, xwm->screen->root_visual, XCB_CW_EVENT_MASK, (uint32_t[]){ XCB_EVENT_MASK_SUBSTRUCTURE_NOTIFY | XCB_EVENT_MASK_PROPERTY_CHANGE } ); if (atom == xwm->atoms[CLIPBOARD]) { xcb_set_selection_owner(xwm->xcb_conn, selection->window, xwm->atoms[CLIPBOARD_MANAGER], XCB_TIME_CURRENT_TIME); } else { assert(atom == xwm->atoms[PRIMARY]); } } uint32_t mask = XCB_XFIXES_SELECTION_EVENT_MASK_SET_SELECTION_OWNER | XCB_XFIXES_SELECTION_EVENT_MASK_SELECTION_WINDOW_DESTROY | XCB_XFIXES_SELECTION_EVENT_MASK_SELECTION_CLIENT_CLOSE; xcb_xfixes_select_selection_input(xwm->xcb_conn, selection->window, selection->atom, mask); } void xwm_selection_finish(struct wlr_xwm_selection *selection) { if (!selection) { return; } struct wlr_xwm_selection_transfer *outgoing, *tmp; wl_list_for_each_safe(outgoing, tmp, &selection->outgoing, link) { wlr_log(WLR_INFO, "destroyed pending transfer %p", outgoing); xwm_selection_transfer_destroy_outgoing(outgoing); } struct wlr_xwm_selection_transfer *incoming; wl_list_for_each_safe(incoming, tmp, &selection->incoming, link) { xwm_selection_transfer_destroy(incoming); } xcb_destroy_window(selection->xwm->xcb_conn, selection->window); } static void xwm_selection_set_owner(struct wlr_xwm_selection *selection, bool set) { if (set) { xcb_set_selection_owner(selection->xwm->xcb_conn, selection->window, selection->atom, XCB_TIME_CURRENT_TIME); xcb_flush(selection->xwm->xcb_conn); } else { if (selection->owner == selection->window) { xcb_set_selection_owner(selection->xwm->xcb_conn, XCB_WINDOW_NONE, selection->atom, selection->timestamp); xcb_flush(selection->xwm->xcb_conn); } } } static void handle_seat_set_selection(struct wl_listener *listener, void *data) { struct wlr_seat *seat = data; struct wlr_xwm *xwm = wl_container_of(listener, xwm, seat_set_selection); struct wlr_data_source *source = seat->selection_source; if (source != NULL && data_source_is_xwayland(source)) { return; } xwm_selection_set_owner(&xwm->clipboard_selection, source != NULL); } static void handle_seat_set_primary_selection(struct wl_listener *listener, void *data) { struct wlr_seat *seat = data; struct wlr_xwm *xwm = wl_container_of(listener, xwm, seat_set_primary_selection); struct wlr_primary_selection_source *source = seat->primary_selection_source; if (source != NULL && primary_selection_source_is_xwayland(source)) { return; } xwm_selection_set_owner(&xwm->primary_selection, source != NULL); } static void seat_handle_start_drag(struct wl_listener *listener, void *data) { struct wlr_xwm *xwm = wl_container_of(listener, xwm, seat_start_drag); struct wlr_drag *drag = data; xwm_selection_set_owner(&xwm->dnd_selection, drag != NULL); xwm_seat_handle_start_drag(xwm, drag); } void xwm_set_seat(struct wlr_xwm *xwm, struct wlr_seat *seat) { if (xwm->seat != NULL) { wl_list_remove(&xwm->seat_set_selection.link); wl_list_remove(&xwm->seat_set_primary_selection.link); wl_list_remove(&xwm->seat_start_drag.link); xwm->seat = NULL; } if (seat == NULL) { return; } xwm->seat = seat; wl_signal_add(&seat->events.set_selection, &xwm->seat_set_selection); xwm->seat_set_selection.notify = handle_seat_set_selection; wl_signal_add(&seat->events.set_primary_selection, &xwm->seat_set_primary_selection); xwm->seat_set_primary_selection.notify = handle_seat_set_primary_selection; wl_signal_add(&seat->events.start_drag, &xwm->seat_start_drag); xwm->seat_start_drag.notify = seat_handle_start_drag; handle_seat_set_selection(&xwm->seat_set_selection, seat); handle_seat_set_primary_selection(&xwm->seat_set_primary_selection, seat); }