mirror of
https://github.com/hyprwm/wlroots-hyprland.git
synced 2024-11-29 07:55:59 +01:00
Merge branch 'master' into feature/xdg-popup
This commit is contained in:
commit
4f848000af
27 changed files with 204 additions and 66 deletions
|
@ -469,7 +469,8 @@ static void wlr_drm_connector_transform(struct wlr_output *output,
|
||||||
}
|
}
|
||||||
|
|
||||||
static bool wlr_drm_connector_set_cursor(struct wlr_output *output,
|
static bool wlr_drm_connector_set_cursor(struct wlr_output *output,
|
||||||
const uint8_t *buf, int32_t stride, uint32_t width, uint32_t height) {
|
const uint8_t *buf, int32_t stride, uint32_t width, uint32_t height,
|
||||||
|
int32_t hotspot_x, int32_t hotspot_y) {
|
||||||
struct wlr_drm_connector *conn = (struct wlr_drm_connector *)output;
|
struct wlr_drm_connector *conn = (struct wlr_drm_connector *)output;
|
||||||
struct wlr_drm_backend *drm = conn->drm;
|
struct wlr_drm_backend *drm = conn->drm;
|
||||||
struct wlr_drm_renderer *renderer = &drm->renderer;
|
struct wlr_drm_renderer *renderer = &drm->renderer;
|
||||||
|
@ -534,6 +535,37 @@ static bool wlr_drm_connector_set_cursor(struct wlr_output *output,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
switch (output->transform) {
|
||||||
|
case WL_OUTPUT_TRANSFORM_90:
|
||||||
|
output->cursor.hotspot_x = hotspot_x;
|
||||||
|
output->cursor.hotspot_y = -plane->surf.height + hotspot_y;
|
||||||
|
break;
|
||||||
|
case WL_OUTPUT_TRANSFORM_180:
|
||||||
|
output->cursor.hotspot_x = plane->surf.width - hotspot_x;
|
||||||
|
output->cursor.hotspot_y = plane->surf.height - hotspot_y;
|
||||||
|
break;
|
||||||
|
case WL_OUTPUT_TRANSFORM_270:
|
||||||
|
output->cursor.hotspot_x = -plane->surf.height + hotspot_x;
|
||||||
|
output->cursor.hotspot_y = hotspot_y;
|
||||||
|
break;
|
||||||
|
case WL_OUTPUT_TRANSFORM_FLIPPED:
|
||||||
|
output->cursor.hotspot_x = plane->surf.width - hotspot_x;
|
||||||
|
output->cursor.hotspot_y = hotspot_y;
|
||||||
|
break;
|
||||||
|
case WL_OUTPUT_TRANSFORM_FLIPPED_90:
|
||||||
|
output->cursor.hotspot_x = hotspot_x;
|
||||||
|
output->cursor.hotspot_y = -hotspot_y;
|
||||||
|
break;
|
||||||
|
case WL_OUTPUT_TRANSFORM_FLIPPED_180:
|
||||||
|
output->cursor.hotspot_x = hotspot_x;
|
||||||
|
output->cursor.hotspot_y = plane->surf.height - hotspot_y;
|
||||||
|
break;
|
||||||
|
case WL_OUTPUT_TRANSFORM_FLIPPED_270:
|
||||||
|
output->cursor.hotspot_x = -plane->surf.height + hotspot_x;
|
||||||
|
output->cursor.hotspot_y = plane->surf.width - hotspot_y;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
struct gbm_bo *bo = plane->cursor_bo;
|
struct gbm_bo *bo = plane->cursor_bo;
|
||||||
uint32_t bo_width = gbm_bo_get_width(bo);
|
uint32_t bo_width = gbm_bo_get_width(bo);
|
||||||
uint32_t bo_height = gbm_bo_get_height(bo);
|
uint32_t bo_height = gbm_bo_get_height(bo);
|
||||||
|
@ -581,23 +613,22 @@ static bool wlr_drm_connector_move_cursor(struct wlr_output *output,
|
||||||
|
|
||||||
switch (output->transform) {
|
switch (output->transform) {
|
||||||
case WL_OUTPUT_TRANSFORM_NORMAL:
|
case WL_OUTPUT_TRANSFORM_NORMAL:
|
||||||
|
case WL_OUTPUT_TRANSFORM_FLIPPED:
|
||||||
|
case WL_OUTPUT_TRANSFORM_FLIPPED_180:
|
||||||
// nothing to do
|
// nothing to do
|
||||||
break;
|
break;
|
||||||
case WL_OUTPUT_TRANSFORM_270:
|
case WL_OUTPUT_TRANSFORM_270:
|
||||||
|
case WL_OUTPUT_TRANSFORM_FLIPPED_270:
|
||||||
tmp = x;
|
tmp = x;
|
||||||
x = y;
|
x = y;
|
||||||
y = -(tmp - width);
|
y = -(tmp - width);
|
||||||
break;
|
break;
|
||||||
case WL_OUTPUT_TRANSFORM_90:
|
case WL_OUTPUT_TRANSFORM_90:
|
||||||
|
case WL_OUTPUT_TRANSFORM_FLIPPED_90:
|
||||||
tmp = x;
|
tmp = x;
|
||||||
x = -(y - height);
|
x = -(y - height);
|
||||||
y = tmp;
|
y = tmp;
|
||||||
break;
|
break;
|
||||||
default:
|
|
||||||
// TODO other transformations
|
|
||||||
wlr_log(L_ERROR, "TODO: handle surface to crtc for transformation = %d",
|
|
||||||
output->transform);
|
|
||||||
break;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return drm->iface->crtc_move_cursor(drm, conn->crtc, x, y);
|
return drm->iface->crtc_move_cursor(drm, conn->crtc, x, y);
|
||||||
|
|
|
@ -67,5 +67,6 @@ void handle_keyboard_key(struct libinput_event *event,
|
||||||
wlr_event.state = WLR_KEY_PRESSED;
|
wlr_event.state = WLR_KEY_PRESSED;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
wlr_keyboard_update_state(wlr_dev->keyboard, &wlr_event);
|
wlr_event.update_state = true;
|
||||||
|
wlr_keyboard_notify_key(wlr_dev->keyboard, &wlr_event);
|
||||||
}
|
}
|
||||||
|
|
|
@ -53,7 +53,8 @@ static void wlr_wl_output_transform(struct wlr_output *_output,
|
||||||
}
|
}
|
||||||
|
|
||||||
static bool wlr_wl_output_set_cursor(struct wlr_output *_output,
|
static bool wlr_wl_output_set_cursor(struct wlr_output *_output,
|
||||||
const uint8_t *buf, int32_t stride, uint32_t width, uint32_t height) {
|
const uint8_t *buf, int32_t stride, uint32_t width, uint32_t height,
|
||||||
|
int32_t hotspot_x, int32_t hotspot_y) {
|
||||||
|
|
||||||
struct wlr_wl_backend_output *output = (struct wlr_wl_backend_output *)_output;
|
struct wlr_wl_backend_output *output = (struct wlr_wl_backend_output *)_output;
|
||||||
struct wlr_wl_backend *backend = output->backend;
|
struct wlr_wl_backend *backend = output->backend;
|
||||||
|
@ -110,7 +111,8 @@ static bool wlr_wl_output_set_cursor(struct wlr_output *_output,
|
||||||
wl_surface_damage(output->cursor_surface, 0, 0, width, height);
|
wl_surface_damage(output->cursor_surface, 0, 0, width, height);
|
||||||
wl_surface_commit(output->cursor_surface);
|
wl_surface_commit(output->cursor_surface);
|
||||||
|
|
||||||
wlr_wl_output_update_cursor(output, output->enter_serial);
|
wlr_wl_output_update_cursor(output, output->enter_serial,
|
||||||
|
hotspot_x, hotspot_y);
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -143,10 +145,11 @@ static void wlr_wl_output_destroy(struct wlr_output *_output) {
|
||||||
free(output);
|
free(output);
|
||||||
}
|
}
|
||||||
|
|
||||||
void wlr_wl_output_update_cursor(struct wlr_wl_backend_output *output, uint32_t serial) {
|
void wlr_wl_output_update_cursor(struct wlr_wl_backend_output *output,
|
||||||
|
uint32_t serial, int32_t hotspot_x, int32_t hotspot_y) {
|
||||||
if (output->cursor_surface && output->backend->pointer && serial) {
|
if (output->cursor_surface && output->backend->pointer && serial) {
|
||||||
wl_pointer_set_cursor(output->backend->pointer, serial,
|
wl_pointer_set_cursor(output->backend->pointer, serial,
|
||||||
output->cursor_surface, 0, 0);
|
output->cursor_surface, hotspot_x, hotspot_y);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -24,7 +24,7 @@ static void pointer_handle_enter(void *data, struct wl_pointer *wl_pointer,
|
||||||
assert(output);
|
assert(output);
|
||||||
wlr_wl_pointer->current_output = output;
|
wlr_wl_pointer->current_output = output;
|
||||||
wlr_wl_pointer->current_output->enter_serial = serial;
|
wlr_wl_pointer->current_output->enter_serial = serial;
|
||||||
wlr_wl_output_update_cursor(wlr_wl_pointer->current_output, serial);
|
wlr_wl_output_update_cursor(wlr_wl_pointer->current_output, serial, 0, 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void pointer_handle_leave(void *data, struct wl_pointer *wl_pointer,
|
static void pointer_handle_leave(void *data, struct wl_pointer *wl_pointer,
|
||||||
|
@ -149,13 +149,16 @@ static void keyboard_handle_key(void *data, struct wl_keyboard *wl_keyboard,
|
||||||
wlr_event.state = state;
|
wlr_event.state = state;
|
||||||
wlr_event.time_sec = time / 1000;
|
wlr_event.time_sec = time / 1000;
|
||||||
wlr_event.time_usec = time * 1000;
|
wlr_event.time_usec = time * 1000;
|
||||||
wlr_keyboard_update_state(dev->keyboard, &wlr_event);
|
wlr_keyboard_notify_key(dev->keyboard, &wlr_event);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void keyboard_handle_modifiers(void *data, struct wl_keyboard *wl_keyboard,
|
static void keyboard_handle_modifiers(void *data, struct wl_keyboard *wl_keyboard,
|
||||||
uint32_t serial, uint32_t mods_depressed, uint32_t mods_latched,
|
uint32_t serial, uint32_t mods_depressed, uint32_t mods_latched,
|
||||||
uint32_t mods_locked, uint32_t group) {
|
uint32_t mods_locked, uint32_t group) {
|
||||||
|
struct wlr_input_device *dev = data;
|
||||||
|
assert(dev && dev->keyboard);
|
||||||
|
wlr_keyboard_notify_modifiers(dev->keyboard, mods_depressed, mods_latched,
|
||||||
|
mods_locked, group);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void keyboard_handle_repeat_info(void *data, struct wl_keyboard *wl_keyboard,
|
static void keyboard_handle_repeat_info(void *data, struct wl_keyboard *wl_keyboard,
|
||||||
|
|
|
@ -51,9 +51,11 @@ static bool handle_x11_event(struct wlr_x11_backend *x11, xcb_generic_event_t *e
|
||||||
.keycode = ev->detail - 8,
|
.keycode = ev->detail - 8,
|
||||||
.state = event->response_type == XCB_KEY_PRESS ?
|
.state = event->response_type == XCB_KEY_PRESS ?
|
||||||
WLR_KEY_PRESSED : WLR_KEY_RELEASED,
|
WLR_KEY_PRESSED : WLR_KEY_RELEASED,
|
||||||
|
.update_state = true,
|
||||||
};
|
};
|
||||||
|
|
||||||
wl_signal_emit(&x11->keyboard.events.key, &key);
|
// TODO use xcb-xkb for more precise modifiers state?
|
||||||
|
wlr_keyboard_notify_key(&x11->keyboard, &key);
|
||||||
x11->time = ev->time;
|
x11->time = ev->time;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
|
@ -112,7 +112,8 @@ static void handle_output_add(struct output_state *ostate) {
|
||||||
// TODO the cursor must be set depending on which surface it is displayed
|
// TODO the cursor must be set depending on which surface it is displayed
|
||||||
// over which should happen in the compositor.
|
// over which should happen in the compositor.
|
||||||
if (!wlr_output_set_cursor(wlr_output, image->buffer,
|
if (!wlr_output_set_cursor(wlr_output, image->buffer,
|
||||||
image->width, image->width, image->height)) {
|
image->width, image->width, image->height,
|
||||||
|
image->hotspot_x, image->hotspot_y)) {
|
||||||
wlr_log(L_DEBUG, "Failed to set hardware cursor");
|
wlr_log(L_DEBUG, "Failed to set hardware cursor");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
|
@ -67,7 +67,8 @@ struct wlr_wl_pointer {
|
||||||
};
|
};
|
||||||
|
|
||||||
void wlr_wl_registry_poll(struct wlr_wl_backend *backend);
|
void wlr_wl_registry_poll(struct wlr_wl_backend *backend);
|
||||||
void wlr_wl_output_update_cursor(struct wlr_wl_backend_output *output, uint32_t serial);
|
void wlr_wl_output_update_cursor(struct wlr_wl_backend_output *output,
|
||||||
|
uint32_t serial, int32_t hotspot_x, int32_t hotspot_y);
|
||||||
struct wlr_wl_backend_output *wlr_wl_output_for_surface(
|
struct wlr_wl_backend_output *wlr_wl_output_for_surface(
|
||||||
struct wlr_wl_backend *backend, struct wl_surface *surface);
|
struct wlr_wl_backend *backend, struct wl_surface *surface);
|
||||||
|
|
||||||
|
|
|
@ -81,7 +81,7 @@ struct roots_input {
|
||||||
struct wlr_seat *wl_seat;
|
struct wlr_seat *wl_seat;
|
||||||
|
|
||||||
enum roots_cursor_mode mode;
|
enum roots_cursor_mode mode;
|
||||||
struct roots_view *active_view;
|
struct roots_view *active_view, *last_active_view;
|
||||||
int offs_x, offs_y;
|
int offs_x, offs_y;
|
||||||
int view_x, view_y, view_width, view_height;
|
int view_x, view_y, view_width, view_height;
|
||||||
float view_rotation;
|
float view_rotation;
|
||||||
|
|
|
@ -65,11 +65,13 @@ struct roots_view {
|
||||||
void (*get_input_bounds)(struct roots_view *view, struct wlr_box *box);
|
void (*get_input_bounds)(struct roots_view *view, struct wlr_box *box);
|
||||||
void (*activate)(struct roots_view *view, bool active);
|
void (*activate)(struct roots_view *view, bool active);
|
||||||
void (*resize)(struct roots_view *view, uint32_t width, uint32_t height);
|
void (*resize)(struct roots_view *view, uint32_t width, uint32_t height);
|
||||||
|
void (*close)(struct roots_view *view);
|
||||||
};
|
};
|
||||||
|
|
||||||
void view_get_size(struct roots_view *view, struct wlr_box *box);
|
void view_get_size(struct roots_view *view, struct wlr_box *box);
|
||||||
void view_get_input_bounds(struct roots_view *view, struct wlr_box *box);
|
void view_get_input_bounds(struct roots_view *view, struct wlr_box *box);
|
||||||
void view_activate(struct roots_view *view, bool active);
|
void view_activate(struct roots_view *view, bool active);
|
||||||
void view_resize(struct roots_view *view, uint32_t width, uint32_t height);
|
void view_resize(struct roots_view *view, uint32_t width, uint32_t height);
|
||||||
|
void view_close(struct roots_view *view);
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
|
@ -11,7 +11,10 @@ struct wlr_keyboard_impl {
|
||||||
|
|
||||||
void wlr_keyboard_init(struct wlr_keyboard *keyboard, struct wlr_keyboard_impl *impl);
|
void wlr_keyboard_init(struct wlr_keyboard *keyboard, struct wlr_keyboard_impl *impl);
|
||||||
void wlr_keyboard_destroy(struct wlr_keyboard *keyboard);
|
void wlr_keyboard_destroy(struct wlr_keyboard *keyboard);
|
||||||
void wlr_keyboard_update_state(struct wlr_keyboard *keyboard,
|
void wlr_keyboard_notify_key(struct wlr_keyboard *keyboard,
|
||||||
struct wlr_event_keyboard_key *event);
|
struct wlr_event_keyboard_key *event);
|
||||||
|
void wlr_keyboard_notify_modifiers(struct wlr_keyboard *keyboard,
|
||||||
|
uint32_t mods_depressed, uint32_t mods_latched, uint32_t mods_locked,
|
||||||
|
uint32_t group);
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
|
@ -10,7 +10,8 @@ struct wlr_output_impl {
|
||||||
void (*transform)(struct wlr_output *output,
|
void (*transform)(struct wlr_output *output,
|
||||||
enum wl_output_transform transform);
|
enum wl_output_transform transform);
|
||||||
bool (*set_cursor)(struct wlr_output *output, const uint8_t *buf,
|
bool (*set_cursor)(struct wlr_output *output, const uint8_t *buf,
|
||||||
int32_t stride, uint32_t width, uint32_t height);
|
int32_t stride, uint32_t width, uint32_t height,
|
||||||
|
int32_t hotspot_x, int32_t hotspot_y);
|
||||||
bool (*move_cursor)(struct wlr_output *output, int x, int y);
|
bool (*move_cursor)(struct wlr_output *output, int x, int y);
|
||||||
void (*destroy)(struct wlr_output *output);
|
void (*destroy)(struct wlr_output *output);
|
||||||
void (*make_current)(struct wlr_output *output);
|
void (*make_current)(struct wlr_output *output);
|
||||||
|
|
|
@ -65,6 +65,7 @@ struct wlr_event_keyboard_key {
|
||||||
uint32_t time_sec;
|
uint32_t time_sec;
|
||||||
uint64_t time_usec;
|
uint64_t time_usec;
|
||||||
uint32_t keycode;
|
uint32_t keycode;
|
||||||
|
bool update_state;
|
||||||
enum wlr_key_state state;
|
enum wlr_key_state state;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
|
@ -45,6 +45,7 @@ struct wlr_output {
|
||||||
bool is_sw;
|
bool is_sw;
|
||||||
int32_t x, y;
|
int32_t x, y;
|
||||||
uint32_t width, height;
|
uint32_t width, height;
|
||||||
|
int32_t hotspot_x, hotspot_y;
|
||||||
struct wlr_renderer *renderer;
|
struct wlr_renderer *renderer;
|
||||||
struct wlr_texture *texture;
|
struct wlr_texture *texture;
|
||||||
} cursor;
|
} cursor;
|
||||||
|
@ -58,7 +59,8 @@ bool wlr_output_set_mode(struct wlr_output *output,
|
||||||
void wlr_output_transform(struct wlr_output *output,
|
void wlr_output_transform(struct wlr_output *output,
|
||||||
enum wl_output_transform transform);
|
enum wl_output_transform transform);
|
||||||
bool wlr_output_set_cursor(struct wlr_output *output,
|
bool wlr_output_set_cursor(struct wlr_output *output,
|
||||||
const uint8_t *buf, int32_t stride, uint32_t width, uint32_t height);
|
const uint8_t *buf, int32_t stride, uint32_t width, uint32_t height,
|
||||||
|
int32_t hotspot_x, int32_t hotspot_y);
|
||||||
bool wlr_output_move_cursor(struct wlr_output *output, int x, int y);
|
bool wlr_output_move_cursor(struct wlr_output *output, int x, int y);
|
||||||
void wlr_output_destroy(struct wlr_output *output);
|
void wlr_output_destroy(struct wlr_output *output);
|
||||||
void wlr_output_effective_resolution(struct wlr_output *output,
|
void wlr_output_effective_resolution(struct wlr_output *output,
|
||||||
|
|
|
@ -201,4 +201,9 @@ void wlr_xdg_toplevel_v6_set_fullscreen(struct wlr_xdg_surface_v6 *surface,
|
||||||
void wlr_xdg_toplevel_v6_set_resizing(struct wlr_xdg_surface_v6 *surface,
|
void wlr_xdg_toplevel_v6_set_resizing(struct wlr_xdg_surface_v6 *surface,
|
||||||
bool resizing);
|
bool resizing);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Request that this toplevel surface closes.
|
||||||
|
*/
|
||||||
|
void wlr_xdg_toplevel_v6_send_close(struct wlr_xdg_surface_v6 *surface);
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
|
@ -133,15 +133,19 @@ static void set_view_focus(struct roots_input *input,
|
||||||
if (!view) {
|
if (!view) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
input->last_active_view = view;
|
||||||
|
|
||||||
size_t index = 0;
|
size_t index = 0;
|
||||||
for (size_t i = 0; i < desktop->views->length; ++i) {
|
for (size_t i = 0; i < desktop->views->length; ++i) {
|
||||||
struct roots_view *_view = desktop->views->items[i];
|
struct roots_view *_view = desktop->views->items[i];
|
||||||
view_activate(_view, _view == view);
|
if (_view != view) {
|
||||||
|
view_activate(_view, false);
|
||||||
|
}
|
||||||
if (view == _view) {
|
if (view == _view) {
|
||||||
index = i;
|
index = i;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
view_activate(view, true);
|
||||||
// TODO: list_swap
|
// TODO: list_swap
|
||||||
list_del(desktop->views, index);
|
list_del(desktop->views, index);
|
||||||
list_add(desktop->views, view);
|
list_add(desktop->views, view);
|
||||||
|
|
|
@ -16,6 +16,16 @@
|
||||||
|
|
||||||
void view_destroy(struct roots_view *view) {
|
void view_destroy(struct roots_view *view) {
|
||||||
struct roots_desktop *desktop = view->desktop;
|
struct roots_desktop *desktop = view->desktop;
|
||||||
|
|
||||||
|
struct roots_input *input = desktop->server->input;
|
||||||
|
if (input->active_view == view) {
|
||||||
|
input->active_view = NULL;
|
||||||
|
input->mode = ROOTS_CURSOR_PASSTHROUGH;
|
||||||
|
}
|
||||||
|
if (input->last_active_view == view) {
|
||||||
|
input->last_active_view = NULL;
|
||||||
|
}
|
||||||
|
|
||||||
for (size_t i = 0; i < desktop->views->length; ++i) {
|
for (size_t i = 0; i < desktop->views->length; ++i) {
|
||||||
struct roots_view *_view = desktop->views->items[i];
|
struct roots_view *_view = desktop->views->items[i];
|
||||||
if (view == _view) {
|
if (view == _view) {
|
||||||
|
@ -67,6 +77,12 @@ void view_resize(struct roots_view *view, uint32_t width, uint32_t height) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void view_close(struct roots_view *view) {
|
||||||
|
if (view->close) {
|
||||||
|
view->close(view);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
static struct wlr_subsurface *subsurface_at(struct wlr_surface *surface,
|
static struct wlr_subsurface *subsurface_at(struct wlr_surface *surface,
|
||||||
double sx, double sy, double *sub_x, double *sub_y) {
|
double sx, double sy, double *sub_x, double *sub_y) {
|
||||||
struct wlr_subsurface *subsurface;
|
struct wlr_subsurface *subsurface;
|
||||||
|
|
|
@ -21,19 +21,28 @@ static ssize_t keyboard_pressed_keysym_index(struct roots_keyboard *keyboard,
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static const char *exec_prefix = "exec ";
|
||||||
|
|
||||||
static void keyboard_binding_execute(struct roots_keyboard *keyboard,
|
static void keyboard_binding_execute(struct roots_keyboard *keyboard,
|
||||||
char *command) {
|
const char *command) {
|
||||||
struct roots_server *server = keyboard->input->server;
|
struct roots_server *server = keyboard->input->server;
|
||||||
if (strcmp(command, "exit") == 0) {
|
if (strcmp(command, "exit") == 0) {
|
||||||
wl_display_terminate(server->wl_display);
|
wl_display_terminate(server->wl_display);
|
||||||
} else {
|
} else if (strcmp(command, "close") == 0) {
|
||||||
|
if (keyboard->input->last_active_view != NULL) {
|
||||||
|
view_close(keyboard->input->last_active_view);
|
||||||
|
}
|
||||||
|
} else if (strncmp(exec_prefix, command, strlen(exec_prefix)) == 0) {
|
||||||
|
const char *shell_cmd = command + strlen(exec_prefix);
|
||||||
pid_t pid = fork();
|
pid_t pid = fork();
|
||||||
if (pid < 0) {
|
if (pid < 0) {
|
||||||
wlr_log(L_ERROR, "cannot execute binding command: fork() failed");
|
wlr_log(L_ERROR, "cannot execute binding command: fork() failed");
|
||||||
return;
|
return;
|
||||||
} else if (pid == 0) {
|
} else if (pid == 0) {
|
||||||
execl("/bin/sh", "/bin/sh", "-c", command, (void *)NULL);
|
execl("/bin/sh", "/bin/sh", "-c", shell_cmd, (void *)NULL);
|
||||||
}
|
}
|
||||||
|
} else {
|
||||||
|
wlr_log(L_ERROR, "unknown binding command: %s", command);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -169,7 +169,8 @@ void output_add_notify(struct wl_listener *listener, void *data) {
|
||||||
// TODO the cursor must be set depending on which surface it is displayed
|
// TODO the cursor must be set depending on which surface it is displayed
|
||||||
// over which should happen in the compositor.
|
// over which should happen in the compositor.
|
||||||
if (!wlr_output_set_cursor(wlr_output, image->buffer,
|
if (!wlr_output_set_cursor(wlr_output, image->buffer,
|
||||||
image->width, image->width, image->height)) {
|
image->width, image->width, image->height,
|
||||||
|
image->hotspot_x, image->hotspot_y)) {
|
||||||
wlr_log(L_DEBUG, "Failed to set hardware cursor");
|
wlr_log(L_DEBUG, "Failed to set hardware cursor");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
|
@ -30,6 +30,7 @@ meta-key = Logo
|
||||||
|
|
||||||
# Keybindings
|
# Keybindings
|
||||||
# Maps key combinations with commands to execute
|
# Maps key combinations with commands to execute
|
||||||
# The special command "exit" stops the compositor
|
# Use the prefix "exec" to execute a shell command
|
||||||
[bindings]
|
[bindings]
|
||||||
Logo+q = exit
|
Logo+Shift+e = exit # Stop the compositor
|
||||||
|
Logo+q = close # Close the current view
|
||||||
|
|
|
@ -17,6 +17,12 @@ static void resize(struct roots_view *view, uint32_t width, uint32_t height) {
|
||||||
height);
|
height);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void close(struct roots_view *view) {
|
||||||
|
assert(view->type == ROOTS_WL_SHELL_VIEW);
|
||||||
|
struct wlr_wl_shell_surface *surf = view->wl_shell_surface;
|
||||||
|
wl_client_destroy(surf->client);
|
||||||
|
}
|
||||||
|
|
||||||
static void handle_request_move(struct wl_listener *listener, void *data) {
|
static void handle_request_move(struct wl_listener *listener, void *data) {
|
||||||
struct roots_wl_shell_surface *roots_surface =
|
struct roots_wl_shell_surface *roots_surface =
|
||||||
wl_container_of(listener, roots_surface, request_move);
|
wl_container_of(listener, roots_surface, request_move);
|
||||||
|
@ -88,6 +94,7 @@ void handle_wl_shell_surface(struct wl_listener *listener, void *data) {
|
||||||
view->roots_wl_shell_surface = roots_surface;
|
view->roots_wl_shell_surface = roots_surface;
|
||||||
view->wlr_surface = surface->surface;
|
view->wlr_surface = surface->surface;
|
||||||
view->resize = resize;
|
view->resize = resize;
|
||||||
|
view->close = close;
|
||||||
view->desktop = desktop;
|
view->desktop = desktop;
|
||||||
roots_surface->view = view;
|
roots_surface->view = view;
|
||||||
list_add(desktop->views, view);
|
list_add(desktop->views, view);
|
||||||
|
|
|
@ -33,6 +33,14 @@ static void resize(struct roots_view *view, uint32_t width, uint32_t height) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void close(struct roots_view *view) {
|
||||||
|
assert(view->type == ROOTS_XDG_SHELL_V6_VIEW);
|
||||||
|
struct wlr_xdg_surface_v6 *surf = view->xdg_surface_v6;
|
||||||
|
if (surf->role == WLR_XDG_SURFACE_V6_ROLE_TOPLEVEL) {
|
||||||
|
wlr_xdg_toplevel_v6_send_close(surf);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
static void handle_request_move(struct wl_listener *listener, void *data) {
|
static void handle_request_move(struct wl_listener *listener, void *data) {
|
||||||
struct roots_xdg_surface_v6 *roots_xdg_surface =
|
struct roots_xdg_surface_v6 *roots_xdg_surface =
|
||||||
wl_container_of(listener, roots_xdg_surface, request_move);
|
wl_container_of(listener, roots_xdg_surface, request_move);
|
||||||
|
@ -114,6 +122,7 @@ void handle_xdg_shell_v6_surface(struct wl_listener *listener, void *data) {
|
||||||
view->get_size = get_size;
|
view->get_size = get_size;
|
||||||
view->activate = activate;
|
view->activate = activate;
|
||||||
view->resize = resize;
|
view->resize = resize;
|
||||||
|
view->close = close;
|
||||||
view->desktop = desktop;
|
view->desktop = desktop;
|
||||||
roots_surface->view = view;
|
roots_surface->view = view;
|
||||||
list_add(desktop->views, view);
|
list_add(desktop->views, view);
|
||||||
|
|
|
@ -9,6 +9,16 @@
|
||||||
#include "rootston/desktop.h"
|
#include "rootston/desktop.h"
|
||||||
#include "rootston/server.h"
|
#include "rootston/server.h"
|
||||||
|
|
||||||
|
static void activate(struct roots_view *view, bool active) {
|
||||||
|
assert(view->type == ROOTS_XWAYLAND_VIEW);
|
||||||
|
if (active) {
|
||||||
|
wlr_xwayland_surface_activate(view->desktop->xwayland,
|
||||||
|
view->xwayland_surface);
|
||||||
|
} else {
|
||||||
|
wlr_xwayland_surface_activate(view->desktop->xwayland, NULL);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
static void resize(struct roots_view *view, uint32_t width, uint32_t height) {
|
static void resize(struct roots_view *view, uint32_t width, uint32_t height) {
|
||||||
assert(view->type == ROOTS_XWAYLAND_VIEW);
|
assert(view->type == ROOTS_XWAYLAND_VIEW);
|
||||||
struct wlr_xwayland_surface *xwayland_surface = view->xwayland_surface;
|
struct wlr_xwayland_surface *xwayland_surface = view->xwayland_surface;
|
||||||
|
@ -16,6 +26,11 @@ static void resize(struct roots_view *view, uint32_t width, uint32_t height) {
|
||||||
xwayland_surface->x, xwayland_surface->y, width, height);
|
xwayland_surface->x, xwayland_surface->y, width, height);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void close(struct roots_view *view) {
|
||||||
|
assert(view->type == ROOTS_XWAYLAND_VIEW);
|
||||||
|
wlr_xwayland_surface_close(view->desktop->xwayland, view->xwayland_surface);
|
||||||
|
}
|
||||||
|
|
||||||
static void handle_destroy(struct wl_listener *listener, void *data) {
|
static void handle_destroy(struct wl_listener *listener, void *data) {
|
||||||
struct roots_xwayland_surface *roots_surface =
|
struct roots_xwayland_surface *roots_surface =
|
||||||
wl_container_of(listener, roots_surface, destroy);
|
wl_container_of(listener, roots_surface, destroy);
|
||||||
|
@ -38,11 +53,6 @@ static void handle_request_configure(struct wl_listener *listener, void *data) {
|
||||||
xwayland_surface, event->x, event->y, event->width, event->height);
|
xwayland_surface, event->x, event->y, event->width, event->height);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void activate(struct roots_view *view, bool active) {
|
|
||||||
wlr_xwayland_surface_activate(view->desktop->xwayland,
|
|
||||||
view->xwayland_surface);
|
|
||||||
}
|
|
||||||
|
|
||||||
void handle_xwayland_surface(struct wl_listener *listener, void *data) {
|
void handle_xwayland_surface(struct wl_listener *listener, void *data) {
|
||||||
struct roots_desktop *desktop =
|
struct roots_desktop *desktop =
|
||||||
wl_container_of(listener, desktop, xwayland_surface);
|
wl_container_of(listener, desktop, xwayland_surface);
|
||||||
|
@ -78,6 +88,7 @@ void handle_xwayland_surface(struct wl_listener *listener, void *data) {
|
||||||
view->desktop = desktop;
|
view->desktop = desktop;
|
||||||
view->activate = activate;
|
view->activate = activate;
|
||||||
view->resize = resize;
|
view->resize = resize;
|
||||||
|
view->close = close;
|
||||||
roots_surface->view = view;
|
roots_surface->view = view;
|
||||||
list_add(desktop->views, view);
|
list_add(desktop->views, view);
|
||||||
}
|
}
|
||||||
|
|
|
@ -131,15 +131,6 @@ static struct wlr_cursor_device *get_cursor_device(struct wlr_cursor *cur,
|
||||||
static void wlr_cursor_warp_unchecked(struct wlr_cursor *cur,
|
static void wlr_cursor_warp_unchecked(struct wlr_cursor *cur,
|
||||||
double x, double y) {
|
double x, double y) {
|
||||||
assert(cur->state->layout);
|
assert(cur->state->layout);
|
||||||
int hotspot_x = 0;
|
|
||||||
int hotspot_y = 0;
|
|
||||||
|
|
||||||
if (cur->state->xcursor && cur->state->xcursor->image_count > 0) {
|
|
||||||
struct wlr_xcursor_image *image = cur->state->xcursor->images[0];
|
|
||||||
hotspot_x = image->hotspot_x;
|
|
||||||
hotspot_y = image->hotspot_y;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
struct wlr_output_layout_output *l_output;
|
struct wlr_output_layout_output *l_output;
|
||||||
wl_list_for_each(l_output, &cur->state->layout->outputs, link) {
|
wl_list_for_each(l_output, &cur->state->layout->outputs, link) {
|
||||||
|
@ -148,8 +139,9 @@ static void wlr_cursor_warp_unchecked(struct wlr_cursor *cur,
|
||||||
|
|
||||||
wlr_output_layout_output_coords(cur->state->layout,
|
wlr_output_layout_output_coords(cur->state->layout,
|
||||||
l_output->output, &output_x, &output_y);
|
l_output->output, &output_x, &output_y);
|
||||||
wlr_output_move_cursor(l_output->output, output_x - hotspot_x,
|
wlr_output_move_cursor(l_output->output,
|
||||||
output_y - hotspot_y);
|
output_x - l_output->output->cursor.hotspot_x,
|
||||||
|
output_y - l_output->output->cursor.hotspot_y);
|
||||||
}
|
}
|
||||||
|
|
||||||
cur->x = x;
|
cur->x = x;
|
||||||
|
|
|
@ -45,11 +45,21 @@ static void keyboard_modifier_update(struct wlr_keyboard *keyboard) {
|
||||||
wl_signal_emit(&keyboard->events.modifiers, keyboard);
|
wl_signal_emit(&keyboard->events.modifiers, keyboard);
|
||||||
}
|
}
|
||||||
|
|
||||||
void wlr_keyboard_update_state(struct wlr_keyboard *keyboard,
|
void wlr_keyboard_notify_modifiers(struct wlr_keyboard *keyboard,
|
||||||
|
uint32_t mods_depressed, uint32_t mods_latched, uint32_t mods_locked,
|
||||||
|
uint32_t group) {
|
||||||
|
xkb_state_update_mask(keyboard->xkb_state, mods_depressed, mods_latched,
|
||||||
|
mods_locked, 0, 0, group);
|
||||||
|
keyboard_modifier_update(keyboard);
|
||||||
|
}
|
||||||
|
|
||||||
|
void wlr_keyboard_notify_key(struct wlr_keyboard *keyboard,
|
||||||
struct wlr_event_keyboard_key *event) {
|
struct wlr_event_keyboard_key *event) {
|
||||||
uint32_t keycode = event->keycode + 8;
|
if (event->update_state) {
|
||||||
xkb_state_update_key(keyboard->xkb_state, keycode,
|
uint32_t keycode = event->keycode + 8;
|
||||||
event->state == WLR_KEY_PRESSED ? XKB_KEY_DOWN : XKB_KEY_UP);
|
xkb_state_update_key(keyboard->xkb_state, keycode,
|
||||||
|
event->state == WLR_KEY_PRESSED ? XKB_KEY_DOWN : XKB_KEY_UP);
|
||||||
|
}
|
||||||
keyboard_led_update(keyboard);
|
keyboard_led_update(keyboard);
|
||||||
keyboard_modifier_update(keyboard);
|
keyboard_modifier_update(keyboard);
|
||||||
wl_signal_emit(&keyboard->events.key, event);
|
wl_signal_emit(&keyboard->events.key, event);
|
||||||
|
|
|
@ -19,8 +19,8 @@ static void wl_output_send_to_resource(struct wl_resource *resource) {
|
||||||
const uint32_t version = wl_resource_get_version(resource);
|
const uint32_t version = wl_resource_get_version(resource);
|
||||||
if (version >= WL_OUTPUT_GEOMETRY_SINCE_VERSION) {
|
if (version >= WL_OUTPUT_GEOMETRY_SINCE_VERSION) {
|
||||||
wl_output_send_geometry(resource, 0, 0, // TODO: get position from layout?
|
wl_output_send_geometry(resource, 0, 0, // TODO: get position from layout?
|
||||||
output->phys_width, output->phys_height, output->subpixel,
|
output->phys_width, output->phys_height, output->subpixel,
|
||||||
output->make, output->model, output->transform);
|
output->make, output->model, output->transform);
|
||||||
}
|
}
|
||||||
if (version >= WL_OUTPUT_MODE_SINCE_VERSION) {
|
if (version >= WL_OUTPUT_MODE_SINCE_VERSION) {
|
||||||
for (size_t i = 0; i < output->modes->length; ++i) {
|
for (size_t i = 0; i < output->modes->length; ++i) {
|
||||||
|
@ -31,7 +31,13 @@ static void wl_output_send_to_resource(struct wl_resource *resource) {
|
||||||
flags |= WL_OUTPUT_MODE_CURRENT;
|
flags |= WL_OUTPUT_MODE_CURRENT;
|
||||||
}
|
}
|
||||||
wl_output_send_mode(resource, flags,
|
wl_output_send_mode(resource, flags,
|
||||||
mode->width, mode->height, mode->refresh);
|
mode->width, mode->height, mode->refresh);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (output->modes->length == 0) {
|
||||||
|
// Output has no mode, send the current width/height
|
||||||
|
wl_output_send_mode(resource, WL_OUTPUT_MODE_CURRENT,
|
||||||
|
output->width, output->height, 0);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (version >= WL_OUTPUT_SCALE_SINCE_VERSION) {
|
if (version >= WL_OUTPUT_SCALE_SINCE_VERSION) {
|
||||||
|
@ -125,9 +131,11 @@ void wlr_output_transform(struct wlr_output *output,
|
||||||
}
|
}
|
||||||
|
|
||||||
bool wlr_output_set_cursor(struct wlr_output *output,
|
bool wlr_output_set_cursor(struct wlr_output *output,
|
||||||
const uint8_t *buf, int32_t stride, uint32_t width, uint32_t height) {
|
const uint8_t *buf, int32_t stride, uint32_t width, uint32_t height,
|
||||||
|
int32_t hotspot_x, int32_t hotspot_y) {
|
||||||
if (output->impl->set_cursor
|
if (output->impl->set_cursor
|
||||||
&& output->impl->set_cursor(output, buf, stride, width, height)) {
|
&& output->impl->set_cursor(output, buf, stride, width, height,
|
||||||
|
hotspot_x, hotspot_y)) {
|
||||||
output->cursor.is_sw = false;
|
output->cursor.is_sw = false;
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
@ -137,6 +145,8 @@ bool wlr_output_set_cursor(struct wlr_output *output,
|
||||||
output->cursor.is_sw = true;
|
output->cursor.is_sw = true;
|
||||||
output->cursor.width = width;
|
output->cursor.width = width;
|
||||||
output->cursor.height = height;
|
output->cursor.height = height;
|
||||||
|
output->cursor.hotspot_x = hotspot_x;
|
||||||
|
output->cursor.hotspot_y = hotspot_y;
|
||||||
|
|
||||||
if (!output->cursor.renderer) {
|
if (!output->cursor.renderer) {
|
||||||
/* NULL egl is okay given that we are only using pixel buffers */
|
/* NULL egl is okay given that we are only using pixel buffers */
|
||||||
|
|
|
@ -1296,3 +1296,8 @@ void wlr_xdg_toplevel_v6_set_resizing(struct wlr_xdg_surface_v6 *surface,
|
||||||
|
|
||||||
wlr_xdg_surface_v6_schedule_configure(surface, false);
|
wlr_xdg_surface_v6_schedule_configure(surface, false);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void wlr_xdg_toplevel_v6_send_close(struct wlr_xdg_surface_v6 *surface) {
|
||||||
|
assert(surface->role == WLR_XDG_SURFACE_V6_ROLE_TOPLEVEL);
|
||||||
|
zxdg_toplevel_v6_send_close(surface->toplevel_state->resource);
|
||||||
|
}
|
||||||
|
|
|
@ -705,19 +705,25 @@ static void xcb_init_wm(struct wlr_xwm *xwm) {
|
||||||
void wlr_xwayland_surface_activate(struct wlr_xwayland *wlr_xwayland,
|
void wlr_xwayland_surface_activate(struct wlr_xwayland *wlr_xwayland,
|
||||||
struct wlr_xwayland_surface *surface) {
|
struct wlr_xwayland_surface *surface) {
|
||||||
struct wlr_xwm *xwm = wlr_xwayland->xwm;
|
struct wlr_xwm *xwm = wlr_xwayland->xwm;
|
||||||
xcb_client_message_event_t m = {0};
|
if (surface) {
|
||||||
m.response_type = XCB_CLIENT_MESSAGE;
|
xcb_client_message_event_t m = {0};
|
||||||
m.format = 32;
|
m.response_type = XCB_CLIENT_MESSAGE;
|
||||||
m.window = surface->window_id;
|
m.format = 32;
|
||||||
m.type = xwm->atoms[WM_PROTOCOLS];
|
m.window = surface->window_id;
|
||||||
m.data.data32[0] = xwm->atoms[WM_TAKE_FOCUS];
|
m.type = xwm->atoms[WM_PROTOCOLS];
|
||||||
m.data.data32[1] = XCB_TIME_CURRENT_TIME;
|
m.data.data32[0] = xwm->atoms[WM_TAKE_FOCUS];
|
||||||
xcb_send_event_checked(xwm->xcb_conn, 0, surface->window_id,
|
m.data.data32[1] = XCB_TIME_CURRENT_TIME;
|
||||||
XCB_EVENT_MASK_SUBSTRUCTURE_REDIRECT, (char*)&m);
|
xcb_send_event_checked(xwm->xcb_conn, 0, surface->window_id,
|
||||||
xcb_set_input_focus_checked(xwm->xcb_conn, XCB_INPUT_FOCUS_POINTER_ROOT,
|
XCB_EVENT_MASK_SUBSTRUCTURE_REDIRECT, (char*)&m);
|
||||||
surface->window_id, XCB_CURRENT_TIME);
|
xcb_set_input_focus_checked(xwm->xcb_conn, XCB_INPUT_FOCUS_POINTER_ROOT,
|
||||||
xcb_configure_window_checked(xwm->xcb_conn, surface->window_id,
|
surface->window_id, XCB_CURRENT_TIME);
|
||||||
XCB_CONFIG_WINDOW_STACK_MODE, (uint32_t[]){XCB_STACK_MODE_ABOVE});
|
xcb_configure_window_checked(xwm->xcb_conn, surface->window_id,
|
||||||
|
XCB_CONFIG_WINDOW_STACK_MODE, (uint32_t[]){XCB_STACK_MODE_ABOVE});
|
||||||
|
} else {
|
||||||
|
wlr_log(L_DEBUG, "Deactivating xwayland");
|
||||||
|
xcb_set_input_focus_checked(xwm->xcb_conn, XCB_INPUT_FOCUS_NONE,
|
||||||
|
-1, XCB_CURRENT_TIME);
|
||||||
|
}
|
||||||
xcb_flush(xwm->xcb_conn);
|
xcb_flush(xwm->xcb_conn);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -735,6 +741,7 @@ void wlr_xwayland_surface_configure(struct wlr_xwayland *wlr_xwayland,
|
||||||
XCB_CONFIG_WINDOW_BORDER_WIDTH;
|
XCB_CONFIG_WINDOW_BORDER_WIDTH;
|
||||||
uint32_t values[] = {x, y, width, height, 0};
|
uint32_t values[] = {x, y, width, height, 0};
|
||||||
xcb_configure_window(xwm->xcb_conn, surface->window_id, mask, values);
|
xcb_configure_window(xwm->xcb_conn, surface->window_id, mask, values);
|
||||||
|
xcb_flush(xwm->xcb_conn);
|
||||||
}
|
}
|
||||||
|
|
||||||
void wlr_xwayland_surface_close(struct wlr_xwayland *wlr_xwayland,
|
void wlr_xwayland_surface_close(struct wlr_xwayland *wlr_xwayland,
|
||||||
|
|
Loading…
Reference in a new issue