| 
									
										
										
										
											2021-02-23 17:36:32 +01:00
										 |  |  | #include <drm_fourcc.h>
 | 
					
						
							| 
									
										
										
										
											2018-03-19 23:16:29 +01:00
										 |  |  | #include <limits.h>
 | 
					
						
							|  |  |  | #include <math.h>
 | 
					
						
							| 
									
										
										
										
											2017-08-16 11:51:22 -04:00
										 |  |  | #include <stdio.h>
 | 
					
						
							|  |  |  | #include <stdlib.h>
 | 
					
						
							|  |  |  | #include <string.h>
 | 
					
						
							|  |  |  | #include <strings.h>
 | 
					
						
							| 
									
										
										
										
											2018-03-19 23:16:29 +01:00
										 |  |  | #include <time.h>
 | 
					
						
							| 
									
										
										
										
											2017-08-16 11:51:22 -04:00
										 |  |  | #include <unistd.h>
 | 
					
						
							| 
									
										
										
										
											2019-07-27 11:53:54 +03:00
										 |  |  | #include <wayland-server-core.h>
 | 
					
						
							| 
									
										
										
										
											2017-08-16 11:51:22 -04:00
										 |  |  | #include <wlr/backend.h>
 | 
					
						
							|  |  |  | #include <wlr/backend/session.h>
 | 
					
						
							| 
									
										
										
										
											2021-09-24 09:34:51 -04:00
										 |  |  | #include <wlr/render/allocator.h>
 | 
					
						
							| 
									
										
										
										
											2018-03-19 23:16:29 +01:00
										 |  |  | #include <wlr/render/wlr_renderer.h>
 | 
					
						
							|  |  |  | #include <wlr/types/wlr_keyboard.h>
 | 
					
						
							| 
									
										
										
										
											2018-04-29 17:32:00 -04:00
										 |  |  | #include <wlr/types/wlr_input_device.h>
 | 
					
						
							| 
									
										
										
										
											2017-08-16 11:51:22 -04:00
										 |  |  | #include <wlr/types/wlr_output_layout.h>
 | 
					
						
							|  |  |  | #include <wlr/types/wlr_output.h>
 | 
					
						
							| 
									
										
										
										
											2021-07-01 16:36:01 -04:00
										 |  |  | #include <wlr/util/box.h>
 | 
					
						
							| 
									
										
										
										
											2018-03-19 23:16:29 +01:00
										 |  |  | #include <wlr/util/log.h>
 | 
					
						
							|  |  |  | #include <xkbcommon/xkbcommon.h>
 | 
					
						
							| 
									
										
										
										
											2018-04-29 17:32:00 -04:00
										 |  |  | #include "cat.h"
 | 
					
						
							| 
									
										
										
										
											2017-08-16 11:51:22 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | struct sample_state { | 
					
						
							| 
									
										
										
										
											2018-04-29 17:32:00 -04:00
										 |  |  | 	struct wl_display *display; | 
					
						
							|  |  |  | 	struct wl_listener new_output; | 
					
						
							|  |  |  | 	struct wl_listener new_input; | 
					
						
							| 
									
										
										
										
											2017-08-16 11:51:22 -04:00
										 |  |  | 	struct wlr_renderer *renderer; | 
					
						
							| 
									
										
										
										
											2021-09-24 09:34:51 -04:00
										 |  |  | 	struct wlr_allocator *allocator; | 
					
						
							| 
									
										
										
										
											2017-08-16 11:51:22 -04:00
										 |  |  | 	struct wlr_texture *cat_texture; | 
					
						
							|  |  |  | 	struct wlr_output_layout *layout; | 
					
						
							|  |  |  | 	float x_offs, y_offs; | 
					
						
							|  |  |  | 	float x_vel, y_vel; | 
					
						
							| 
									
										
										
										
											2017-08-31 23:43:02 -04:00
										 |  |  | 	struct timespec ts_last; | 
					
						
							| 
									
										
										
										
											2017-08-16 11:51:22 -04:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-29 17:32:00 -04:00
										 |  |  | struct sample_output { | 
					
						
							|  |  |  | 	struct sample_state *sample; | 
					
						
							|  |  |  | 	struct wlr_output *output; | 
					
						
							|  |  |  | 	struct wl_listener frame; | 
					
						
							|  |  |  | 	struct wl_listener destroy; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct sample_keyboard { | 
					
						
							|  |  |  | 	struct sample_state *sample; | 
					
						
							| 
									
										
										
										
											2022-06-20 16:57:29 +02:00
										 |  |  | 	struct wlr_keyboard *wlr_keyboard; | 
					
						
							| 
									
										
										
										
											2018-04-29 17:32:00 -04:00
										 |  |  | 	struct wl_listener key; | 
					
						
							|  |  |  | 	struct wl_listener destroy; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-31 23:43:02 -04:00
										 |  |  | static void animate_cat(struct sample_state *sample, | 
					
						
							|  |  |  | 		struct wlr_output *output) { | 
					
						
							|  |  |  | 	struct timespec ts; | 
					
						
							|  |  |  | 	clock_gettime(CLOCK_MONOTONIC, &ts); | 
					
						
							|  |  |  | 	long ms = (ts.tv_sec - sample->ts_last.tv_sec) * 1000 + | 
					
						
							|  |  |  | 		(ts.tv_nsec - sample->ts_last.tv_nsec) / 1000000; | 
					
						
							|  |  |  | 	// how many seconds have passed since the last time we animated
 | 
					
						
							|  |  |  | 	float seconds = ms / 1000.0f; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (seconds > 0.1f) { | 
					
						
							|  |  |  | 		// XXX when we switch vt, the rendering loop stops so try to detect
 | 
					
						
							|  |  |  | 		// that and pause when it happens.
 | 
					
						
							|  |  |  | 		seconds = 0.0f; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// check for collisions and bounce
 | 
					
						
							|  |  |  | 	bool ur_collision = !wlr_output_layout_output_at(sample->layout, | 
					
						
							|  |  |  | 			sample->x_offs + 128, sample->y_offs); | 
					
						
							|  |  |  | 	bool ul_collision = !wlr_output_layout_output_at(sample->layout, | 
					
						
							|  |  |  | 			sample->x_offs, sample->y_offs); | 
					
						
							|  |  |  | 	bool ll_collision = !wlr_output_layout_output_at(sample->layout, | 
					
						
							|  |  |  | 			sample->x_offs, sample->y_offs + 128); | 
					
						
							|  |  |  | 	bool lr_collision = !wlr_output_layout_output_at(sample->layout, | 
					
						
							|  |  |  | 			sample->x_offs + 128, sample->y_offs + 128); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (ur_collision && ul_collision && ll_collision && lr_collision) { | 
					
						
							|  |  |  | 		// oops we went off the screen somehow
 | 
					
						
							|  |  |  | 		struct wlr_output_layout_output *l_output = | 
					
						
							|  |  |  | 			wlr_output_layout_get(sample->layout, output); | 
					
						
							|  |  |  | 		sample->x_offs = l_output->x + 20; | 
					
						
							|  |  |  | 		sample->y_offs = l_output->y + 20; | 
					
						
							|  |  |  | 	} else if (ur_collision && ul_collision) { | 
					
						
							|  |  |  | 		sample->y_vel = fabs(sample->y_vel); | 
					
						
							|  |  |  | 	} else if (lr_collision && ll_collision) { | 
					
						
							|  |  |  | 		sample->y_vel = -fabs(sample->y_vel); | 
					
						
							|  |  |  | 	} else if (ll_collision && ul_collision) { | 
					
						
							|  |  |  | 		sample->x_vel = fabs(sample->x_vel); | 
					
						
							|  |  |  | 	} else if (ur_collision && lr_collision) { | 
					
						
							|  |  |  | 		sample->x_vel = -fabs(sample->x_vel); | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		if (ur_collision || lr_collision) { | 
					
						
							|  |  |  | 			sample->x_vel = -fabs(sample->x_vel); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		if (ul_collision || ll_collision) { | 
					
						
							|  |  |  | 			sample->x_vel = fabs(sample->x_vel); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		if (ul_collision || ur_collision) { | 
					
						
							|  |  |  | 			sample->y_vel = fabs(sample->y_vel); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		if (ll_collision || lr_collision) { | 
					
						
							|  |  |  | 			sample->y_vel = -fabs(sample->y_vel); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	sample->x_offs += sample->x_vel * seconds; | 
					
						
							|  |  |  | 	sample->y_offs += sample->y_vel * seconds; | 
					
						
							|  |  |  | 	sample->ts_last = ts; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-11-20 00:45:19 +01:00
										 |  |  | static void output_frame_notify(struct wl_listener *listener, void *data) { | 
					
						
							| 
									
										
										
										
											2018-04-29 17:32:00 -04:00
										 |  |  | 	struct sample_output *output = wl_container_of(listener, output, frame); | 
					
						
							|  |  |  | 	struct sample_state *sample = output->sample; | 
					
						
							| 
									
										
										
										
											2017-08-16 11:51:22 -04:00
										 |  |  | 	struct wlr_output *wlr_output = output->output; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-06-22 15:48:35 +02:00
										 |  |  | 	struct wlr_output_state output_state; | 
					
						
							|  |  |  | 	wlr_output_state_init(&output_state); | 
					
						
							| 
									
										
										
										
											2023-06-02 10:25:07 +01:00
										 |  |  | 	struct wlr_render_pass *pass = wlr_output_begin_render_pass(wlr_output, &output_state, NULL, NULL); | 
					
						
							| 
									
										
										
										
											2023-04-17 19:44:33 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	wlr_render_pass_add_rect(pass, &(struct wlr_render_rect_options){ | 
					
						
							|  |  |  | 		.box = { .width = wlr_output->width, .height = wlr_output->height }, | 
					
						
							|  |  |  | 		.color = { 0.25, 0.25, 0.25, 1 }, | 
					
						
							|  |  |  | 	}); | 
					
						
							| 
									
										
										
										
											2017-08-16 11:51:22 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-31 23:43:02 -04:00
										 |  |  | 	animate_cat(sample, output->output); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-12-31 12:49:06 +01:00
										 |  |  | 	struct wlr_box box = { | 
					
						
							|  |  |  | 		.x = sample->x_offs, .y = sample->y_offs, | 
					
						
							|  |  |  | 		.width = 128, .height = 128, | 
					
						
							|  |  |  | 	}; | 
					
						
							|  |  |  | 	if (wlr_output_layout_intersects(sample->layout, output->output, &box)) { | 
					
						
							| 
									
										
										
										
											2017-08-17 10:12:36 -04:00
										 |  |  | 		// transform global coordinates to local coordinates
 | 
					
						
							| 
									
										
										
										
											2017-08-24 11:46:40 -04:00
										 |  |  | 		double local_x = sample->x_offs; | 
					
						
							|  |  |  | 		double local_y = sample->y_offs; | 
					
						
							| 
									
										
										
										
											2017-08-30 10:43:48 -04:00
										 |  |  | 		wlr_output_layout_output_coords(sample->layout, output->output, | 
					
						
							|  |  |  | 			&local_x, &local_y); | 
					
						
							| 
									
										
										
										
											2017-08-16 11:51:22 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-04-17 19:44:33 +02:00
										 |  |  | 		wlr_render_pass_add_texture(pass, &(struct wlr_render_texture_options){ | 
					
						
							|  |  |  | 			.texture = sample->cat_texture, | 
					
						
							|  |  |  | 			.dst_box = box, | 
					
						
							|  |  |  | 		}); | 
					
						
							| 
									
										
										
										
											2017-08-16 15:06:38 -04:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2017-08-16 11:51:22 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-04-17 19:44:33 +02:00
										 |  |  | 	wlr_render_pass_submit(pass); | 
					
						
							|  |  |  | 	wlr_output_commit_state(wlr_output, &output_state); | 
					
						
							|  |  |  | 	wlr_output_state_finish(&output_state); | 
					
						
							| 
									
										
										
										
											2017-08-16 11:51:22 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-29 17:32:00 -04:00
										 |  |  | static void update_velocities(struct sample_state *sample, | 
					
						
							|  |  |  | 		float x_diff, float y_diff) { | 
					
						
							|  |  |  | 	sample->x_vel += x_diff; | 
					
						
							|  |  |  | 	sample->y_vel += y_diff; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2017-08-30 10:39:22 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-11-20 00:45:19 +01:00
										 |  |  | static void output_remove_notify(struct wl_listener *listener, void *data) { | 
					
						
							| 
									
										
										
										
											2018-04-29 17:32:00 -04:00
										 |  |  | 	struct sample_output *sample_output = wl_container_of(listener, sample_output, destroy); | 
					
						
							|  |  |  | 	struct sample_state *sample = sample_output->sample; | 
					
						
							|  |  |  | 	wlr_output_layout_remove(sample->layout, sample_output->output); | 
					
						
							|  |  |  | 	wl_list_remove(&sample_output->frame.link); | 
					
						
							|  |  |  | 	wl_list_remove(&sample_output->destroy.link); | 
					
						
							|  |  |  | 	free(sample_output); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2017-08-30 10:39:22 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-11-20 00:45:19 +01:00
										 |  |  | static void new_output_notify(struct wl_listener *listener, void *data) { | 
					
						
							| 
									
										
										
										
											2018-04-29 17:32:00 -04:00
										 |  |  | 	struct wlr_output *output = data; | 
					
						
							|  |  |  | 	struct sample_state *sample = wl_container_of(listener, sample, new_output); | 
					
						
							| 
									
										
										
										
											2021-09-24 09:34:51 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	wlr_output_init_render(output, sample->allocator, sample->renderer); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-10-03 01:51:07 -04:00
										 |  |  | 	struct sample_output *sample_output = calloc(1, sizeof(*sample_output)); | 
					
						
							| 
									
										
										
										
											2018-04-29 17:32:00 -04:00
										 |  |  | 	wlr_output_layout_add_auto(sample->layout, output); | 
					
						
							|  |  |  | 	sample_output->output = output; | 
					
						
							|  |  |  | 	sample_output->sample = sample; | 
					
						
							|  |  |  | 	wl_signal_add(&output->events.frame, &sample_output->frame); | 
					
						
							|  |  |  | 	sample_output->frame.notify = output_frame_notify; | 
					
						
							|  |  |  | 	wl_signal_add(&output->events.destroy, &sample_output->destroy); | 
					
						
							|  |  |  | 	sample_output->destroy.notify = output_remove_notify; | 
					
						
							| 
									
										
										
										
											2020-02-08 19:09:41 +13:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-06-26 20:07:36 +02:00
										 |  |  | 	struct wlr_output_state state; | 
					
						
							|  |  |  | 	wlr_output_state_init(&state); | 
					
						
							|  |  |  | 	wlr_output_state_set_enabled(&state, true); | 
					
						
							| 
									
										
										
										
											2021-04-19 17:25:13 +02:00
										 |  |  | 	struct wlr_output_mode *mode = wlr_output_preferred_mode(output); | 
					
						
							|  |  |  | 	if (mode != NULL) { | 
					
						
							| 
									
										
										
										
											2023-06-12 20:16:25 -04:00
										 |  |  | 		wlr_output_state_set_mode(&state, mode); | 
					
						
							| 
									
										
										
										
											2021-04-19 17:25:13 +02:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2023-06-26 20:07:36 +02:00
										 |  |  | 	wlr_output_commit_state(output, &state); | 
					
						
							|  |  |  | 	wlr_output_state_finish(&state); | 
					
						
							| 
									
										
										
										
											2017-08-30 13:14:52 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-11-20 00:45:19 +01:00
										 |  |  | static void keyboard_key_notify(struct wl_listener *listener, void *data) { | 
					
						
							| 
									
										
										
										
											2018-04-29 17:32:00 -04:00
										 |  |  | 	struct sample_keyboard *keyboard = wl_container_of(listener, keyboard, key); | 
					
						
							|  |  |  | 	struct sample_state *sample = keyboard->sample; | 
					
						
							| 
									
										
										
										
											2022-03-09 15:05:12 -05:00
										 |  |  | 	struct wlr_keyboard_key_event *event = data; | 
					
						
							| 
									
										
										
										
											2018-04-29 17:32:00 -04:00
										 |  |  | 	uint32_t keycode = event->keycode + 8; | 
					
						
							|  |  |  | 	const xkb_keysym_t *syms; | 
					
						
							| 
									
										
										
										
											2022-06-20 16:57:29 +02:00
										 |  |  | 	int nsyms = xkb_state_key_get_syms(keyboard->wlr_keyboard->xkb_state, | 
					
						
							| 
									
										
										
										
											2018-04-29 17:32:00 -04:00
										 |  |  | 			keycode, &syms); | 
					
						
							|  |  |  | 	for (int i = 0; i < nsyms; i++) { | 
					
						
							|  |  |  | 		xkb_keysym_t sym = syms[i]; | 
					
						
							|  |  |  | 		if (sym == XKB_KEY_Escape) { | 
					
						
							| 
									
										
										
										
											2018-04-29 20:16:29 -04:00
										 |  |  | 			wl_display_terminate(sample->display); | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2018-04-29 17:32:00 -04:00
										 |  |  | 		// NOTE: It may be better to simply refer to our key state during each frame
 | 
					
						
							|  |  |  | 		// and make this change in pixels/sec^2
 | 
					
						
							|  |  |  | 		// Also, key repeat
 | 
					
						
							|  |  |  | 		int delta = 75; | 
					
						
							| 
									
										
										
										
											2020-10-21 17:21:23 +02:00
										 |  |  | 		if (event->state == WL_KEYBOARD_KEY_STATE_PRESSED) { | 
					
						
							| 
									
										
										
										
											2018-04-29 17:32:00 -04:00
										 |  |  | 			switch (sym) { | 
					
						
							|  |  |  | 			case XKB_KEY_Left: | 
					
						
							|  |  |  | 				update_velocities(sample, -delta, 0); | 
					
						
							|  |  |  | 				break; | 
					
						
							|  |  |  | 			case XKB_KEY_Right: | 
					
						
							|  |  |  | 				update_velocities(sample, delta, 0); | 
					
						
							|  |  |  | 				break; | 
					
						
							|  |  |  | 			case XKB_KEY_Up: | 
					
						
							|  |  |  | 				update_velocities(sample, 0, -delta); | 
					
						
							|  |  |  | 				break; | 
					
						
							|  |  |  | 			case XKB_KEY_Down: | 
					
						
							|  |  |  | 				update_velocities(sample, 0, delta); | 
					
						
							|  |  |  | 				break; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2017-08-16 11:51:22 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-11-20 00:45:19 +01:00
										 |  |  | static void keyboard_destroy_notify(struct wl_listener *listener, void *data) { | 
					
						
							| 
									
										
										
										
											2018-04-29 17:32:00 -04:00
										 |  |  | 	struct sample_keyboard *keyboard = wl_container_of(listener, keyboard, destroy); | 
					
						
							|  |  |  | 	wl_list_remove(&keyboard->destroy.link); | 
					
						
							|  |  |  | 	wl_list_remove(&keyboard->key.link); | 
					
						
							|  |  |  | 	free(keyboard); | 
					
						
							| 
									
										
										
										
											2017-08-16 11:51:22 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-11-20 00:45:19 +01:00
										 |  |  | static void new_input_notify(struct wl_listener *listener, void *data) { | 
					
						
							| 
									
										
										
										
											2018-04-29 17:32:00 -04:00
										 |  |  | 	struct wlr_input_device *device = data; | 
					
						
							|  |  |  | 	struct sample_state *sample = wl_container_of(listener, sample, new_input); | 
					
						
							|  |  |  | 	switch (device->type) { | 
					
						
							| 
									
										
										
										
											2018-04-29 20:16:29 -04:00
										 |  |  | 	case WLR_INPUT_DEVICE_KEYBOARD:; | 
					
						
							| 
									
										
										
										
											2023-10-03 01:51:07 -04:00
										 |  |  | 		struct sample_keyboard *keyboard = calloc(1, sizeof(*keyboard)); | 
					
						
							| 
									
										
										
										
											2022-06-20 16:57:29 +02:00
										 |  |  | 		keyboard->wlr_keyboard = wlr_keyboard_from_input_device(device); | 
					
						
							| 
									
										
										
										
											2018-04-29 17:32:00 -04:00
										 |  |  | 		keyboard->sample = sample; | 
					
						
							|  |  |  | 		wl_signal_add(&device->events.destroy, &keyboard->destroy); | 
					
						
							|  |  |  | 		keyboard->destroy.notify = keyboard_destroy_notify; | 
					
						
							| 
									
										
										
										
											2022-06-20 16:57:29 +02:00
										 |  |  | 		wl_signal_add(&keyboard->wlr_keyboard->events.key, &keyboard->key); | 
					
						
							| 
									
										
										
										
											2018-04-29 17:32:00 -04:00
										 |  |  | 		keyboard->key.notify = keyboard_key_notify; | 
					
						
							| 
									
										
										
										
											2018-04-29 20:16:29 -04:00
										 |  |  | 		struct xkb_context *context = xkb_context_new(XKB_CONTEXT_NO_FLAGS); | 
					
						
							| 
									
										
										
										
											2018-04-29 17:32:00 -04:00
										 |  |  | 		if (!context) { | 
					
						
							| 
									
										
										
										
											2018-07-09 22:49:54 +01:00
										 |  |  | 			wlr_log(WLR_ERROR, "Failed to create XKB context"); | 
					
						
							| 
									
										
										
										
											2018-04-29 17:32:00 -04:00
										 |  |  | 			exit(1); | 
					
						
							| 
									
										
										
										
											2017-08-16 11:51:22 -04:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2021-04-16 09:05:39 +02:00
										 |  |  | 		struct xkb_keymap *keymap = xkb_keymap_new_from_names(context, NULL, | 
					
						
							| 
									
										
										
										
											2018-09-02 21:16:24 +05:30
										 |  |  | 			XKB_KEYMAP_COMPILE_NO_FLAGS); | 
					
						
							|  |  |  | 		if (!keymap) { | 
					
						
							|  |  |  | 			wlr_log(WLR_ERROR, "Failed to create XKB keymap"); | 
					
						
							|  |  |  | 			exit(1); | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2022-06-20 16:57:29 +02:00
										 |  |  | 		wlr_keyboard_set_keymap(keyboard->wlr_keyboard, keymap); | 
					
						
							| 
									
										
										
										
											2018-09-02 21:16:24 +05:30
										 |  |  | 		xkb_keymap_unref(keymap); | 
					
						
							| 
									
										
										
										
											2018-04-29 17:32:00 -04:00
										 |  |  | 		xkb_context_unref(context); | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 	default: | 
					
						
							|  |  |  | 		break; | 
					
						
							| 
									
										
										
										
											2017-08-16 11:51:22 -04:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-29 17:32:00 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-16 11:51:22 -04:00
										 |  |  | int main(int argc, char *argv[]) { | 
					
						
							| 
									
										
										
										
											2018-07-09 22:49:54 +01:00
										 |  |  | 	wlr_log_init(WLR_DEBUG, NULL); | 
					
						
							| 
									
										
										
										
											2018-04-29 17:32:00 -04:00
										 |  |  | 	struct wl_display *display = wl_display_create(); | 
					
						
							|  |  |  | 	struct sample_state state = { | 
					
						
							|  |  |  | 		.x_vel = 500, | 
					
						
							|  |  |  | 		.y_vel = 500, | 
					
						
							|  |  |  | 		.display = display, | 
					
						
							|  |  |  | 	}; | 
					
						
							| 
									
										
										
										
											2017-08-16 11:51:22 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-12 08:24:08 +02:00
										 |  |  | 	state.layout = wlr_output_layout_create(display); | 
					
						
							| 
									
										
										
										
											2017-08-31 23:43:02 -04:00
										 |  |  | 	clock_gettime(CLOCK_MONOTONIC, &state.ts_last); | 
					
						
							| 
									
										
										
										
											2017-08-16 11:51:22 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-23 14:17:39 +01:00
										 |  |  | 	struct wlr_backend *wlr = wlr_backend_autocreate(wl_display_get_event_loop(display), NULL); | 
					
						
							| 
									
										
										
										
											2018-04-29 17:32:00 -04:00
										 |  |  | 	if (!wlr) { | 
					
						
							|  |  |  | 		exit(1); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	wl_signal_add(&wlr->events.new_output, &state.new_output); | 
					
						
							|  |  |  | 	state.new_output.notify = new_output_notify; | 
					
						
							|  |  |  | 	wl_signal_add(&wlr->events.new_input, &state.new_input); | 
					
						
							|  |  |  | 	state.new_input.notify = new_input_notify; | 
					
						
							| 
									
										
										
										
											2017-08-16 11:51:22 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-24 09:34:51 -04:00
										 |  |  | 	state.renderer = wlr_renderer_autocreate(wlr); | 
					
						
							| 
									
										
										
										
											2018-03-24 18:30:28 -04:00
										 |  |  | 	state.cat_texture = wlr_texture_from_pixels(state.renderer, | 
					
						
							| 
									
										
										
										
											2021-02-23 17:36:32 +01:00
										 |  |  | 		DRM_FORMAT_ABGR8888, cat_tex.width * 4, cat_tex.width, cat_tex.height, | 
					
						
							| 
									
										
										
										
											2018-03-24 18:30:28 -04:00
										 |  |  | 		cat_tex.pixel_data); | 
					
						
							| 
									
										
										
										
											2017-08-16 11:51:22 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-24 09:34:51 -04:00
										 |  |  | 	state.allocator = wlr_allocator_autocreate(wlr, state.renderer); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-29 17:32:00 -04:00
										 |  |  | 	if (!wlr_backend_start(wlr)) { | 
					
						
							| 
									
										
										
										
											2018-07-09 22:49:54 +01:00
										 |  |  | 		wlr_log(WLR_ERROR, "Failed to start backend"); | 
					
						
							| 
									
										
										
										
											2018-04-29 17:32:00 -04:00
										 |  |  | 		wlr_backend_destroy(wlr); | 
					
						
							| 
									
										
										
										
											2017-09-24 14:12:56 -04:00
										 |  |  | 		exit(1); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2018-04-29 17:32:00 -04:00
										 |  |  | 	wl_display_run(display); | 
					
						
							| 
									
										
										
										
											2017-08-16 11:51:22 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	wlr_texture_destroy(state.cat_texture); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-05-30 08:34:49 +01:00
										 |  |  | 	wl_display_destroy(state.display); | 
					
						
							| 
									
										
										
										
											2017-08-16 11:51:22 -04:00
										 |  |  | } |