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