Add pointer support to layer shell example
This commit is contained in:
parent
dc412c514b
commit
e3eb487dc2
|
@ -7,12 +7,17 @@
|
||||||
#include <time.h>
|
#include <time.h>
|
||||||
#include <unistd.h>
|
#include <unistd.h>
|
||||||
#include <wayland-client.h>
|
#include <wayland-client.h>
|
||||||
|
#include <wayland-cursor.h>
|
||||||
#include <wayland-egl.h>
|
#include <wayland-egl.h>
|
||||||
#include <wlr/render/egl.h>
|
#include <wlr/render/egl.h>
|
||||||
#include <wlr/util/log.h>
|
#include <wlr/util/log.h>
|
||||||
#include "wlr-layer-shell-unstable-v1-client-protocol.h"
|
#include "wlr-layer-shell-unstable-v1-client-protocol.h"
|
||||||
|
|
||||||
static struct wl_compositor *compositor = NULL;
|
static struct wl_compositor *compositor = NULL;
|
||||||
|
static struct wl_seat *seat = NULL;
|
||||||
|
static struct wl_shm *shm = NULL;
|
||||||
|
static struct wl_pointer *pointer = NULL;
|
||||||
|
//static struct wl_keyboard *keyboard = NULL;
|
||||||
static struct zwlr_layer_shell_v1 *layer_shell = NULL;
|
static struct zwlr_layer_shell_v1 *layer_shell = NULL;
|
||||||
struct zwlr_layer_surface_v1 *layer_surface;
|
struct zwlr_layer_surface_v1 *layer_surface;
|
||||||
static struct wl_output *wl_output = NULL;
|
static struct wl_output *wl_output = NULL;
|
||||||
|
@ -32,6 +37,12 @@ static double alpha = 1.0;
|
||||||
static bool run_display = true;
|
static bool run_display = true;
|
||||||
static bool animate = false;
|
static bool animate = false;
|
||||||
static double frame = 0;
|
static double frame = 0;
|
||||||
|
static int cur_x = -1, cur_y = -1;
|
||||||
|
static int buttons = 0;
|
||||||
|
|
||||||
|
struct wl_cursor_theme *cursor_theme;
|
||||||
|
struct wl_cursor_image *cursor_image;
|
||||||
|
struct wl_surface *cursor_surface;
|
||||||
|
|
||||||
static struct {
|
static struct {
|
||||||
struct timespec last_frame;
|
struct timespec last_frame;
|
||||||
|
@ -62,6 +73,7 @@ static void draw(void) {
|
||||||
(ts.tv_nsec - demo.last_frame.tv_nsec) / 1000000;
|
(ts.tv_nsec - demo.last_frame.tv_nsec) / 1000000;
|
||||||
int inc = (demo.dec + 1) % 3;
|
int inc = (demo.dec + 1) % 3;
|
||||||
|
|
||||||
|
if (!buttons) {
|
||||||
demo.color[inc] += ms / 2000.0f;
|
demo.color[inc] += ms / 2000.0f;
|
||||||
demo.color[demo.dec] -= ms / 2000.0f;
|
demo.color[demo.dec] -= ms / 2000.0f;
|
||||||
|
|
||||||
|
@ -70,6 +82,7 @@ static void draw(void) {
|
||||||
demo.color[demo.dec] = 0.0f;
|
demo.color[demo.dec] = 0.0f;
|
||||||
demo.dec = inc;
|
demo.dec = inc;
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if (animate) {
|
if (animate) {
|
||||||
frame += ms / 50.0;
|
frame += ms / 50.0;
|
||||||
|
@ -83,9 +96,21 @@ static void draw(void) {
|
||||||
}
|
}
|
||||||
|
|
||||||
glViewport(0, 0, width, height);
|
glViewport(0, 0, width, height);
|
||||||
|
if (buttons) {
|
||||||
|
glClearColor(1, 1, 1, alpha);
|
||||||
|
} else {
|
||||||
glClearColor(demo.color[0], demo.color[1], demo.color[2], alpha);
|
glClearColor(demo.color[0], demo.color[1], demo.color[2], alpha);
|
||||||
|
}
|
||||||
glClear(GL_COLOR_BUFFER_BIT);
|
glClear(GL_COLOR_BUFFER_BIT);
|
||||||
|
|
||||||
|
if (cur_x != -1 && cur_y != -1) {
|
||||||
|
glEnable(GL_SCISSOR_TEST);
|
||||||
|
glScissor(cur_x, height - cur_y, 5, 5);
|
||||||
|
glClearColor(0, 0, 0, 1);
|
||||||
|
glClear(GL_COLOR_BUFFER_BIT);
|
||||||
|
glDisable(GL_SCISSOR_TEST);
|
||||||
|
}
|
||||||
|
|
||||||
frame_callback = wl_surface_frame(wl_surface);
|
frame_callback = wl_surface_frame(wl_surface);
|
||||||
wl_callback_add_listener(frame_callback, &frame_listener, NULL);
|
wl_callback_add_listener(frame_callback, &frame_listener, NULL);
|
||||||
|
|
||||||
|
@ -119,11 +144,101 @@ struct zwlr_layer_surface_v1_listener layer_surface_listener = {
|
||||||
.closed = layer_surface_closed,
|
.closed = layer_surface_closed,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
static void wl_pointer_enter(void *data, struct wl_pointer *wl_pointer,
|
||||||
|
uint32_t serial, struct wl_surface *surface,
|
||||||
|
wl_fixed_t surface_x, wl_fixed_t surface_y) {
|
||||||
|
wl_surface_attach(cursor_surface,
|
||||||
|
wl_cursor_image_get_buffer(cursor_image), 0, 0);
|
||||||
|
wl_pointer_set_cursor(wl_pointer, serial, cursor_surface,
|
||||||
|
cursor_image->hotspot_x, cursor_image->hotspot_y);
|
||||||
|
wl_surface_commit(cursor_surface);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void wl_pointer_leave(void *data, struct wl_pointer *wl_pointer,
|
||||||
|
uint32_t serial, struct wl_surface *surface) {
|
||||||
|
cur_x = cur_y = -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void wl_pointer_motion(void *data, struct wl_pointer *wl_pointer,
|
||||||
|
uint32_t time, wl_fixed_t surface_x, wl_fixed_t surface_y) {
|
||||||
|
cur_x = (int)wl_fixed_to_double(surface_x);
|
||||||
|
cur_y = (int)wl_fixed_to_double(surface_y);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void wl_pointer_button(void *data, struct wl_pointer *wl_pointer,
|
||||||
|
uint32_t serial, uint32_t time, uint32_t button, uint32_t state) {
|
||||||
|
if (state == WL_POINTER_BUTTON_STATE_PRESSED) {
|
||||||
|
buttons++;
|
||||||
|
} else {
|
||||||
|
buttons--;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static void wl_pointer_axis(void *data, struct wl_pointer *wl_pointer,
|
||||||
|
uint32_t time, uint32_t axis, wl_fixed_t value) {
|
||||||
|
// Who cares
|
||||||
|
}
|
||||||
|
|
||||||
|
static void wl_pointer_frame(void *data, struct wl_pointer *wl_pointer) {
|
||||||
|
// Who cares
|
||||||
|
}
|
||||||
|
|
||||||
|
static void wl_pointer_axis_source(void *data, struct wl_pointer *wl_pointer,
|
||||||
|
uint32_t axis_source) {
|
||||||
|
// Who cares
|
||||||
|
}
|
||||||
|
|
||||||
|
static void wl_pointer_axis_stop(void *data, struct wl_pointer *wl_pointer,
|
||||||
|
uint32_t time, uint32_t axis) {
|
||||||
|
// Who cares
|
||||||
|
}
|
||||||
|
|
||||||
|
static void wl_pointer_axis_discrete(void *data, struct wl_pointer *wl_pointer,
|
||||||
|
uint32_t axis, int32_t discrete) {
|
||||||
|
// Who cares
|
||||||
|
}
|
||||||
|
|
||||||
|
struct wl_pointer_listener pointer_listener = {
|
||||||
|
.enter = wl_pointer_enter,
|
||||||
|
.leave = wl_pointer_leave,
|
||||||
|
.motion = wl_pointer_motion,
|
||||||
|
.button = wl_pointer_button,
|
||||||
|
.axis = wl_pointer_axis,
|
||||||
|
.frame = wl_pointer_frame,
|
||||||
|
.axis_source = wl_pointer_axis_source,
|
||||||
|
.axis_stop = wl_pointer_axis_stop,
|
||||||
|
.axis_discrete = wl_pointer_axis_discrete,
|
||||||
|
};
|
||||||
|
|
||||||
|
static void seat_handle_capabilities(void *data, struct wl_seat *wl_seat,
|
||||||
|
enum wl_seat_capability caps) {
|
||||||
|
if ((caps & WL_SEAT_CAPABILITY_POINTER)) {
|
||||||
|
pointer = wl_seat_get_pointer(wl_seat);
|
||||||
|
wl_pointer_add_listener(pointer, &pointer_listener, NULL);
|
||||||
|
}
|
||||||
|
if ((caps & WL_SEAT_CAPABILITY_KEYBOARD)) {
|
||||||
|
// TODO
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static void seat_handle_name(void *data, struct wl_seat *wl_seat,
|
||||||
|
const char *name) {
|
||||||
|
// Who cares
|
||||||
|
}
|
||||||
|
|
||||||
|
const struct wl_seat_listener seat_listener = {
|
||||||
|
.capabilities = seat_handle_capabilities,
|
||||||
|
.name = seat_handle_name,
|
||||||
|
};
|
||||||
|
|
||||||
static void handle_global(void *data, struct wl_registry *registry,
|
static void handle_global(void *data, struct wl_registry *registry,
|
||||||
uint32_t name, const char *interface, uint32_t version) {
|
uint32_t name, const char *interface, uint32_t version) {
|
||||||
if (strcmp(interface, "wl_compositor") == 0) {
|
if (strcmp(interface, "wl_compositor") == 0) {
|
||||||
compositor = wl_registry_bind(registry, name,
|
compositor = wl_registry_bind(registry, name,
|
||||||
&wl_compositor_interface, 1);
|
&wl_compositor_interface, 1);
|
||||||
|
} else if (strcmp(interface, wl_shm_interface.name) == 0) {
|
||||||
|
shm = wl_registry_bind(registry, name,
|
||||||
|
&wl_shm_interface, 1);
|
||||||
} else if (strcmp(interface, "wl_output") == 0) {
|
} else if (strcmp(interface, "wl_output") == 0) {
|
||||||
if (output == 0 && !wl_output) {
|
if (output == 0 && !wl_output) {
|
||||||
wl_output = wl_registry_bind(registry, name,
|
wl_output = wl_registry_bind(registry, name,
|
||||||
|
@ -131,6 +246,10 @@ static void handle_global(void *data, struct wl_registry *registry,
|
||||||
} else {
|
} else {
|
||||||
output--;
|
output--;
|
||||||
}
|
}
|
||||||
|
} else if (strcmp(interface, wl_seat_interface.name) == 0) {
|
||||||
|
seat = wl_registry_bind(registry, name,
|
||||||
|
&wl_seat_interface, 1);
|
||||||
|
wl_seat_add_listener(seat, &seat_listener, NULL);
|
||||||
} else if (strcmp(interface, zwlr_layer_shell_v1_interface.name) == 0) {
|
} else if (strcmp(interface, zwlr_layer_shell_v1_interface.name) == 0) {
|
||||||
layer_shell = wl_registry_bind(
|
layer_shell = wl_registry_bind(
|
||||||
registry, name, &zwlr_layer_shell_v1_interface, 1);
|
registry, name, &zwlr_layer_shell_v1_interface, 1);
|
||||||
|
@ -250,11 +369,15 @@ int main(int argc, char **argv) {
|
||||||
wl_display_roundtrip(display);
|
wl_display_roundtrip(display);
|
||||||
|
|
||||||
if (compositor == NULL) {
|
if (compositor == NULL) {
|
||||||
fprintf(stderr, "wl-compositor not available\n");
|
fprintf(stderr, "wl_compositor not available\n");
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
if (shm == NULL) {
|
||||||
|
fprintf(stderr, "wl_shm not available\n");
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
if (layer_shell == NULL) {
|
if (layer_shell == NULL) {
|
||||||
fprintf(stderr, "layer-shell not available\n");
|
fprintf(stderr, "layer_shell not available\n");
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
if (wl_output == NULL) {
|
if (wl_output == NULL) {
|
||||||
|
@ -262,6 +385,12 @@ int main(int argc, char **argv) {
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
assert(cursor_theme = wl_cursor_theme_load(NULL, 16, shm));
|
||||||
|
struct wl_cursor *cursor;
|
||||||
|
assert(cursor = wl_cursor_theme_get_cursor(cursor_theme, "left_ptr"));
|
||||||
|
cursor_image = cursor->images[0];
|
||||||
|
assert(cursor_surface = wl_compositor_create_surface(compositor));
|
||||||
|
|
||||||
EGLint attribs[] = { EGL_ALPHA_SIZE, 8, EGL_NONE };
|
EGLint attribs[] = { EGL_ALPHA_SIZE, 8, EGL_NONE };
|
||||||
wlr_egl_init(&egl, EGL_PLATFORM_WAYLAND_EXT, display,
|
wlr_egl_init(&egl, EGL_PLATFORM_WAYLAND_EXT, display,
|
||||||
attribs, WL_SHM_FORMAT_ARGB8888);
|
attribs, WL_SHM_FORMAT_ARGB8888);
|
||||||
|
@ -294,5 +423,7 @@ int main(int argc, char **argv) {
|
||||||
while (wl_display_dispatch(display) != -1 && run_display) {
|
while (wl_display_dispatch(display) != -1 && run_display) {
|
||||||
// This space intentionally left blank
|
// This space intentionally left blank
|
||||||
}
|
}
|
||||||
|
|
||||||
|
wl_cursor_theme_destroy(cursor_theme);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
|
@ -6,6 +6,7 @@ lib_shared = static_library(
|
||||||
)
|
)
|
||||||
|
|
||||||
threads = dependency('threads')
|
threads = dependency('threads')
|
||||||
|
wayland_cursor = dependency('wayland-cursor')
|
||||||
|
|
||||||
executable('simple', 'simple.c', dependencies: wlroots, link_with: lib_shared)
|
executable('simple', 'simple.c', dependencies: wlroots, link_with: lib_shared)
|
||||||
executable('pointer', 'pointer.c', dependencies: wlroots, link_with: lib_shared)
|
executable('pointer', 'pointer.c', dependencies: wlroots, link_with: lib_shared)
|
||||||
|
@ -52,5 +53,5 @@ executable(
|
||||||
executable(
|
executable(
|
||||||
'layer-shell',
|
'layer-shell',
|
||||||
'layer-shell.c',
|
'layer-shell.c',
|
||||||
dependencies: [wayland_client, wlr_protos, wlroots]
|
dependencies: [wayland_cursor, wayland_client, wlr_protos, wlroots]
|
||||||
)
|
)
|
||||||
|
|
|
@ -669,12 +669,13 @@ static struct wlr_surface *layer_surface_at(struct roots_output *output,
|
||||||
double _sx = ox - roots_surface->geo.x;
|
double _sx = ox - roots_surface->geo.x;
|
||||||
double _sy = oy - roots_surface->geo.y;
|
double _sy = oy - roots_surface->geo.y;
|
||||||
struct wlr_box box = {
|
struct wlr_box box = {
|
||||||
.x = 0, .y = 0,
|
.x = roots_surface->geo.x,
|
||||||
.width = roots_surface->geo.width,
|
.y = roots_surface->geo.y,
|
||||||
.height = roots_surface->geo.height,
|
.width = wlr_surface->current->width,
|
||||||
|
.height = wlr_surface->current->height,
|
||||||
};
|
};
|
||||||
// TODO: Test popups/subsurfaces
|
// TODO: Test popups/subsurfaces
|
||||||
if (wlr_box_contains_point(&box, _sx, _sy) &&
|
if (wlr_box_contains_point(&box, ox, oy) &&
|
||||||
pixman_region32_contains_point(&wlr_surface->current->input,
|
pixman_region32_contains_point(&wlr_surface->current->input,
|
||||||
_sx, _sy, NULL)) {
|
_sx, _sy, NULL)) {
|
||||||
*sx = _sx;
|
*sx = _sx;
|
||||||
|
@ -692,7 +693,7 @@ struct wlr_surface *desktop_surface_at(struct roots_desktop *desktop,
|
||||||
struct wlr_output *wlr_output =
|
struct wlr_output *wlr_output =
|
||||||
wlr_output_layout_output_at(desktop->layout, lx, ly);
|
wlr_output_layout_output_at(desktop->layout, lx, ly);
|
||||||
struct roots_output *roots_output;
|
struct roots_output *roots_output;
|
||||||
double ox, oy;
|
double ox = lx, oy = ly;
|
||||||
if (wlr_output) {
|
if (wlr_output) {
|
||||||
roots_output = wlr_output->data;
|
roots_output = wlr_output->data;
|
||||||
wlr_output_layout_output_coords(desktop->layout, wlr_output, &ox, &oy);
|
wlr_output_layout_output_coords(desktop->layout, wlr_output, &ox, &oy);
|
||||||
|
|
Loading…
Reference in New Issue