mirror of
				https://codeberg.org/dwl/dwl.git
				synced 2025-10-31 12:04:16 +00:00 
			
		
		
		
	Merge remote-tracking branch 'upstream/main'
This commit is contained in:
		
						commit
						210bf0389a
					
				
							
								
								
									
										95
									
								
								CHANGELOG.md
									
									
									
									
									
								
							
							
						
						
									
										95
									
								
								CHANGELOG.md
									
									
									
									
									
								
							| @ -1,6 +1,8 @@ | |||||||
| # Changelog | # Changelog | ||||||
| 
 | 
 | ||||||
| * [Unreleased](#unreleased) | * [Unreleased](#unreleased) | ||||||
|  | * [0.7](#0.7) | ||||||
|  | * [0.6](#0.6) | ||||||
| * [0.5](#0.5) | * [0.5](#0.5) | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| @ -14,6 +16,99 @@ | |||||||
| ### Contributors | ### Contributors | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
|  | ## 0.7 | ||||||
|  | 
 | ||||||
|  | This version is just 0.6 with wlroots 0.18 compatibility. | ||||||
|  | 
 | ||||||
|  | ### Added | ||||||
|  | 
 | ||||||
|  | * Add support for the alpha-modifier-v1 protocol ([wlroots!4616][wlroots!4616]). | ||||||
|  | * dwl now will survive GPU resets ([#601][601]). | ||||||
|  | 
 | ||||||
|  | [wlroots!4616]: https://gitlab.freedesktop.org/wlroots/wlroots/-/merge_requests/4616 | ||||||
|  | [601]: https://codeberg.org/dwl/dwl/issues/601 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | ### Contributors | ||||||
|  | 
 | ||||||
|  | Guido Cella | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | ## 0.6 | ||||||
|  | 
 | ||||||
|  | ### Added | ||||||
|  | 
 | ||||||
|  | * Add `rootcolor` to change the default background color ([#544][544]). | ||||||
|  | * Implement the wlr-virtual-pointer-unstable-v1 protocol ([#574][574]). | ||||||
|  | * Implement the pointer-constraints and relative-pointer protocols ([#317][317]) | ||||||
|  | * Implement the wlr-output-power-management protocol ([#599][599]) | ||||||
|  | 
 | ||||||
|  | [544]: https://codeberg.org/dwl/dwl/pulls/544 | ||||||
|  | [574]: https://codeberg.org/dwl/dwl/pulls/574 | ||||||
|  | [317]: https://codeberg.org/dwl/dwl/issues/317 | ||||||
|  | [599]: https://codeberg.org/dwl/dwl/issues/559 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | ### Changed | ||||||
|  | 
 | ||||||
|  | * Keyboards are now managed through keyboard groups ([#549][549]). | ||||||
|  | * Only the first matched keybinding is executed. | ||||||
|  | * Allow toggling the layout before selecting a different one ([#570][570]). | ||||||
|  | * Fullscreen clients are now rendered above wlr_layer_surfaces in the top layer | ||||||
|  |   ([#609][609]). | ||||||
|  | * The default menu was changed from `bemenu-run` to `wmenu-run` ([#553][553]). | ||||||
|  | * The option `sloppyfocus` now replicates the dwm behavior ([#599][599]). | ||||||
|  | * Allow configure position of monitors with negative values. (-1, -1) is | ||||||
|  |   used to auto-configure them ([#635][635]). | ||||||
|  | * dwl now kills the entire process group of `startup_cmd` | ||||||
|  | * The O_NONBLOCK flag is set for stdout. | ||||||
|  | 
 | ||||||
|  | [549]: https://codeberg.org/dwl/dwl/pulls/549 | ||||||
|  | [570]: https://codeberg.org/dwl/dwl/pulls/570 | ||||||
|  | [609]: https://codeberg.org/dwl/dwl/pulls/609 | ||||||
|  | [553]: https://codeberg.org/dwl/dwl/issues/553 | ||||||
|  | [599]: https://codeberg.org/dwl/dwl/pulls/599 | ||||||
|  | [635]: https://codeberg.org/dwl/dwl/pulls/635 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | ### Removed | ||||||
|  | 
 | ||||||
|  | * The SLOC limit is now removed ([#497][497]). | ||||||
|  | 
 | ||||||
|  | [497]: https://codeberg.org/dwl/dwl/pulls/497 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | ### Fixed | ||||||
|  | 
 | ||||||
|  | * Clients not having the correct border color when mapping. | ||||||
|  | * Compliance with the xdg-decoration-unstable-v1 ([#546][546]). | ||||||
|  | * dwl no longer sends negative values in xdg_toplevel.configure events. | ||||||
|  | * Crashes with disabled monitors ([#472][472]). | ||||||
|  | 
 | ||||||
|  | [546]: https://codeberg.org/dwl/dwl/pulls/546 | ||||||
|  | [472]: https://codeberg.org/dwl/dwl/issues/472 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | ### Contributors | ||||||
|  | 
 | ||||||
|  | Ben Jargowsky | ||||||
|  | Benjamin Chausse | ||||||
|  | David Donahue | ||||||
|  | Devin J. Pohly | ||||||
|  | Dima Krasner | ||||||
|  | Emil Miler | ||||||
|  | Forrest Bushstone | ||||||
|  | Guido Cella | ||||||
|  | Peter Hofmann | ||||||
|  | Rutherther | ||||||
|  | Squibid | ||||||
|  | choc | ||||||
|  | fictitiousexistence | ||||||
|  | korei999 | ||||||
|  | sewn | ||||||
|  | thanatos | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
| ## 0.5 | ## 0.5 | ||||||
| 
 | 
 | ||||||
| ### Added | ### Added | ||||||
|  | |||||||
							
								
								
									
										8
									
								
								Makefile
									
									
									
									
									
								
							
							
						
						
									
										8
									
								
								Makefile
									
									
									
									
									
								
							| @ -12,7 +12,7 @@ DWLDEVCFLAGS = -g -pedantic -Wall -Wextra -Wdeclaration-after-statement \ | |||||||
| 	-Wfloat-conversion | 	-Wfloat-conversion | ||||||
| 
 | 
 | ||||||
| # CFLAGS / LDFLAGS
 | # CFLAGS / LDFLAGS
 | ||||||
| PKGS      = wlroots wayland-server xkbcommon libinput pixman-1 fcft $(XLIBS) | PKGS      = wlroots-0.19 wayland-server xkbcommon libinput $(XLIBS) | ||||||
| DWLCFLAGS = `$(PKG_CONFIG) --cflags $(PKGS)` $(DWLCPPFLAGS) $(DWLDEVCFLAGS) $(CFLAGS) | DWLCFLAGS = `$(PKG_CONFIG) --cflags $(PKGS)` $(DWLCPPFLAGS) $(DWLDEVCFLAGS) $(CFLAGS) | ||||||
| LDLIBS    = `$(PKG_CONFIG) --libs $(PKGS)` -lm $(LIBS) | LDLIBS    = `$(PKG_CONFIG) --libs $(PKGS)` -lm $(LIBS) | ||||||
| 
 | 
 | ||||||
| @ -31,13 +31,13 @@ WAYLAND_SCANNER   = `$(PKG_CONFIG) --variable=wayland_scanner wayland-scanner` | |||||||
| WAYLAND_PROTOCOLS = `$(PKG_CONFIG) --variable=pkgdatadir wayland-protocols` | WAYLAND_PROTOCOLS = `$(PKG_CONFIG) --variable=pkgdatadir wayland-protocols` | ||||||
| 
 | 
 | ||||||
| cursor-shape-v1-protocol.h: | cursor-shape-v1-protocol.h: | ||||||
| 	$(WAYLAND_SCANNER) server-header \
 | 	$(WAYLAND_SCANNER) enum-header \
 | ||||||
| 		$(WAYLAND_PROTOCOLS)/staging/cursor-shape/cursor-shape-v1.xml $@ | 		$(WAYLAND_PROTOCOLS)/staging/cursor-shape/cursor-shape-v1.xml $@ | ||||||
| pointer-constraints-unstable-v1-protocol.h: | pointer-constraints-unstable-v1-protocol.h: | ||||||
| 	$(WAYLAND_SCANNER) server-header \
 | 	$(WAYLAND_SCANNER) enum-header \
 | ||||||
| 		$(WAYLAND_PROTOCOLS)/unstable/pointer-constraints/pointer-constraints-unstable-v1.xml $@ | 		$(WAYLAND_PROTOCOLS)/unstable/pointer-constraints/pointer-constraints-unstable-v1.xml $@ | ||||||
| wlr-layer-shell-unstable-v1-protocol.h: | wlr-layer-shell-unstable-v1-protocol.h: | ||||||
| 	$(WAYLAND_SCANNER) server-header \
 | 	$(WAYLAND_SCANNER) enum-header \
 | ||||||
| 		protocols/wlr-layer-shell-unstable-v1.xml $@ | 		protocols/wlr-layer-shell-unstable-v1.xml $@ | ||||||
| wlr-output-power-management-unstable-v1-protocol.h: | wlr-output-power-management-unstable-v1-protocol.h: | ||||||
| 	$(WAYLAND_SCANNER) server-header \
 | 	$(WAYLAND_SCANNER) server-header \
 | ||||||
|  | |||||||
| @ -73,8 +73,9 @@ Xwayland (runtime only) | |||||||
| ``` | ``` | ||||||
| 
 | 
 | ||||||
| Simply install these (and their `-devel` versions if your distro has separate | Simply install these (and their `-devel` versions if your distro has separate | ||||||
| development packages) and run `make`.  If you wish to build against a Git | development packages) and run `make`. You need to use the Git version of | ||||||
| version of wlroots, check out the [wlroots-next branch]. | wlroots to build the `main` branch. If you wish to build against a released | ||||||
|  | version of wlroots, use a release or a [0.x branch]. | ||||||
| 
 | 
 | ||||||
| To enable XWayland, you should uncomment its flags in `config.mk`. | To enable XWayland, you should uncomment its flags in `config.mk`. | ||||||
| 
 | 
 | ||||||
| @ -168,7 +169,7 @@ inspiration, and to the various contributors to the project, including: | |||||||
| [#dwl on Libera Chat]: https://web.libera.chat/?channels=#dwl | [#dwl on Libera Chat]: https://web.libera.chat/?channels=#dwl | ||||||
| [Wayland]: https://wayland.freedesktop.org/ | [Wayland]: https://wayland.freedesktop.org/ | ||||||
| [wlroots]: https://gitlab.freedesktop.org/wlroots/wlroots/ | [wlroots]: https://gitlab.freedesktop.org/wlroots/wlroots/ | ||||||
| [wlroots-next branch]: https://codeberg.org/dwl/dwl/src/branch/wlroots-next | [0.x branch]: https://codeberg.org/dwl/dwl/branches | ||||||
| [patches repository]: https://codeberg.org/dwl/dwl-patches | [patches repository]: https://codeberg.org/dwl/dwl-patches | ||||||
| [s6]: https://skarnet.org/software/s6/ | [s6]: https://skarnet.org/software/s6/ | ||||||
| [anopa]: https://jjacky.com/anopa/ | [anopa]: https://jjacky.com/anopa/ | ||||||
|  | |||||||
							
								
								
									
										4
									
								
								client.h
									
									
									
									
									
								
							
							
						
						
									
										4
									
								
								client.h
									
									
									
									
									
								
							| @ -391,8 +391,8 @@ client_wants_focus(Client *c) | |||||||
| { | { | ||||||
| #ifdef XWAYLAND | #ifdef XWAYLAND | ||||||
| 	return client_is_unmanaged(c) | 	return client_is_unmanaged(c) | ||||||
| 		&& wlr_xwayland_or_surface_wants_focus(c->surface.xwayland) | 		&& wlr_xwayland_surface_override_redirect_wants_focus(c->surface.xwayland) | ||||||
| 		&& wlr_xwayland_icccm_input_model(c->surface.xwayland) != WLR_ICCCM_INPUT_MODEL_NONE; | 		&& wlr_xwayland_surface_icccm_input_model(c->surface.xwayland) != WLR_ICCCM_INPUT_MODEL_NONE; | ||||||
| #endif | #endif | ||||||
| 	return 0; | 	return 0; | ||||||
| } | } | ||||||
|  | |||||||
| @ -1,4 +1,4 @@ | |||||||
| _VERSION = 0.5 | _VERSION = 0.8-dev | ||||||
| VERSION  = `git describe --tags --dirty 2>/dev/null || echo $(_VERSION)` | VERSION  = `git describe --tags --dirty 2>/dev/null || echo $(_VERSION)` | ||||||
| 
 | 
 | ||||||
| PKG_CONFIG = pkg-config | PKG_CONFIG = pkg-config | ||||||
|  | |||||||
							
								
								
									
										187
									
								
								dwl.c
									
									
									
									
									
								
							
							
						
						
									
										187
									
								
								dwl.c
									
									
									
									
									
								
							| @ -17,6 +17,7 @@ | |||||||
| #include <wlr/backend/libinput.h> | #include <wlr/backend/libinput.h> | ||||||
| #include <wlr/render/allocator.h> | #include <wlr/render/allocator.h> | ||||||
| #include <wlr/render/wlr_renderer.h> | #include <wlr/render/wlr_renderer.h> | ||||||
|  | #include <wlr/types/wlr_alpha_modifier_v1.h> | ||||||
| #include <wlr/types/wlr_compositor.h> | #include <wlr/types/wlr_compositor.h> | ||||||
| #include <wlr/types/wlr_cursor.h> | #include <wlr/types/wlr_cursor.h> | ||||||
| #include <wlr/types/wlr_cursor_shape_v1.h> | #include <wlr/types/wlr_cursor_shape_v1.h> | ||||||
| @ -283,6 +284,7 @@ static void cleanupmon(struct wl_listener *listener, void *data); | |||||||
| static void closemon(Monitor *m); | static void closemon(Monitor *m); | ||||||
| static void commitlayersurfacenotify(struct wl_listener *listener, void *data); | static void commitlayersurfacenotify(struct wl_listener *listener, void *data); | ||||||
| static void commitnotify(struct wl_listener *listener, void *data); | static void commitnotify(struct wl_listener *listener, void *data); | ||||||
|  | static void commitpopup(struct wl_listener *listener, void *data); | ||||||
| static void createdecoration(struct wl_listener *listener, void *data); | static void createdecoration(struct wl_listener *listener, void *data); | ||||||
| static void createidleinhibitor(struct wl_listener *listener, void *data); | static void createidleinhibitor(struct wl_listener *listener, void *data); | ||||||
| static void createkeyboard(struct wlr_keyboard *keyboard); | static void createkeyboard(struct wlr_keyboard *keyboard); | ||||||
| @ -293,6 +295,7 @@ static void createmon(struct wl_listener *listener, void *data); | |||||||
| static void createnotify(struct wl_listener *listener, void *data); | static void createnotify(struct wl_listener *listener, void *data); | ||||||
| static void createpointer(struct wlr_pointer *pointer); | static void createpointer(struct wlr_pointer *pointer); | ||||||
| static void createpointerconstraint(struct wl_listener *listener, void *data); | static void createpointerconstraint(struct wl_listener *listener, void *data); | ||||||
|  | static void createpopup(struct wl_listener *listener, void *data); | ||||||
| static void cursorconstrain(struct wlr_pointer_constraint_v1 *constraint); | static void cursorconstrain(struct wlr_pointer_constraint_v1 *constraint); | ||||||
| static void cursorframe(struct wl_listener *listener, void *data); | static void cursorframe(struct wl_listener *listener, void *data); | ||||||
| static void cursorwarptohint(void); | static void cursorwarptohint(void); | ||||||
| @ -315,6 +318,7 @@ static void focusmon(const Arg *arg); | |||||||
| static void focusstack(const Arg *arg); | static void focusstack(const Arg *arg); | ||||||
| static Client *focustop(Monitor *m); | static Client *focustop(Monitor *m); | ||||||
| static void fullscreennotify(struct wl_listener *listener, void *data); | static void fullscreennotify(struct wl_listener *listener, void *data); | ||||||
|  | static void gpureset(struct wl_listener *listener, void *data); | ||||||
| static void handlesig(int signo); | static void handlesig(int signo); | ||||||
| static void incnmaster(const Arg *arg); | static void incnmaster(const Arg *arg); | ||||||
| static void inputdevice(struct wl_listener *listener, void *data); | static void inputdevice(struct wl_listener *listener, void *data); | ||||||
| @ -388,6 +392,7 @@ static pid_t child_pid = -1; | |||||||
| static int locked; | static int locked; | ||||||
| static void *exclusive_focus; | static void *exclusive_focus; | ||||||
| static struct wl_display *dpy; | static struct wl_display *dpy; | ||||||
|  | static struct wl_event_loop *event_loop; | ||||||
| static struct wlr_backend *backend; | static struct wlr_backend *backend; | ||||||
| static struct wlr_scene *scene; | static struct wlr_scene *scene; | ||||||
| static struct wlr_scene_tree *layers[NUM_LAYERS]; | static struct wlr_scene_tree *layers[NUM_LAYERS]; | ||||||
| @ -538,7 +543,7 @@ arrange(Monitor *m) | |||||||
| 	/* We move all clients (except fullscreen and unmanaged) to LyrTile while
 | 	/* We move all clients (except fullscreen and unmanaged) to LyrTile while
 | ||||||
| 	 * in floating layout to avoid "real" floating clients be always on top */ | 	 * in floating layout to avoid "real" floating clients be always on top */ | ||||||
| 	wl_list_for_each(c, &clients, link) { | 	wl_list_for_each(c, &clients, link) { | ||||||
| 		if (c->mon != m || c->isfullscreen) | 		if (c->mon != m || c->scene->node.parent == layers[LyrFS]) | ||||||
| 			continue; | 			continue; | ||||||
| 
 | 
 | ||||||
| 		wlr_scene_node_reparent(&c->scene->node, | 		wlr_scene_node_reparent(&c->scene->node, | ||||||
| @ -631,7 +636,7 @@ axisnotify(struct wl_listener *listener, void *data) | |||||||
| 	/* Notify the client with pointer focus of the axis event. */ | 	/* Notify the client with pointer focus of the axis event. */ | ||||||
| 	wlr_seat_pointer_notify_axis(seat, | 	wlr_seat_pointer_notify_axis(seat, | ||||||
| 			event->time_msec, event->orientation, event->delta, | 			event->time_msec, event->orientation, event->delta, | ||||||
| 			event->delta_discrete, event->source); | 			event->delta_discrete, event->source, event->relative_direction); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| bool | bool | ||||||
| @ -711,7 +716,7 @@ buttonpress(struct wl_listener *listener, void *data) | |||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	switch (event->state) { | 	switch (event->state) { | ||||||
| 	case WLR_BUTTON_PRESSED: | 	case WL_POINTER_BUTTON_STATE_PRESSED: | ||||||
| 		cursor_mode = CurPressed; | 		cursor_mode = CurPressed; | ||||||
| 		selmon = xytomon(cursor->x, cursor->y); | 		selmon = xytomon(cursor->x, cursor->y); | ||||||
| 		if (locked) | 		if (locked) | ||||||
| @ -731,7 +736,7 @@ buttonpress(struct wl_listener *listener, void *data) | |||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
| 		break; | 		break; | ||||||
| 	case WLR_BUTTON_RELEASED: | 	case WL_POINTER_BUTTON_STATE_RELEASED: | ||||||
| 		/* If you released any buttons, we exit interactive move/resize mode. */ | 		/* If you released any buttons, we exit interactive move/resize mode. */ | ||||||
| 		/* TODO should reset to the pointer focus's current setcursor */ | 		/* TODO should reset to the pointer focus's current setcursor */ | ||||||
| 		if (!locked && cursor_mode != CurNormal && cursor_mode != CurPressed) { | 		if (!locked && cursor_mode != CurNormal && cursor_mode != CurPressed) { | ||||||
| @ -789,10 +794,13 @@ cleanup(void) | |||||||
| 		waitpid(child_pid, NULL, 0); | 		waitpid(child_pid, NULL, 0); | ||||||
| 	} | 	} | ||||||
| 	wlr_xcursor_manager_destroy(cursor_mgr); | 	wlr_xcursor_manager_destroy(cursor_mgr); | ||||||
| 	wlr_output_layout_destroy(output_layout); |  | ||||||
| 
 | 
 | ||||||
| 	destroykeyboardgroup(&kb_group->destroy, NULL); | 	destroykeyboardgroup(&kb_group->destroy, NULL); | ||||||
| 
 | 
 | ||||||
|  | 	/* If it's not destroyed manually it will cause a use-after-free of wlr_seat.
 | ||||||
|  | 	 * Destroy it until it's fixed in the wlroots side */ | ||||||
|  | 	wlr_backend_destroy(backend); | ||||||
|  | 
 | ||||||
| 	wl_display_destroy(dpy); | 	wl_display_destroy(dpy); | ||||||
| 	/* Destroy after the wayland display (when the monitors are already destroyed)
 | 	/* Destroy after the wayland display (when the monitors are already destroyed)
 | ||||||
| 	   to avoid destroying them with an invalid scene output. */ | 	   to avoid destroying them with an invalid scene output. */ | ||||||
| @ -864,6 +872,17 @@ commitlayersurfacenotify(struct wl_listener *listener, void *data) | |||||||
| 	LayerSurface *l = wl_container_of(listener, l, surface_commit); | 	LayerSurface *l = wl_container_of(listener, l, surface_commit); | ||||||
| 	struct wlr_layer_surface_v1 *layer_surface = l->layer_surface; | 	struct wlr_layer_surface_v1 *layer_surface = l->layer_surface; | ||||||
| 	struct wlr_scene_tree *scene_layer = layers[layermap[layer_surface->current.layer]]; | 	struct wlr_scene_tree *scene_layer = layers[layermap[layer_surface->current.layer]]; | ||||||
|  | 	struct wlr_layer_surface_v1_state old_state; | ||||||
|  | 
 | ||||||
|  | 	if (l->layer_surface->initial_commit) { | ||||||
|  | 		/* Temporarily set the layer's current state to pending
 | ||||||
|  | 		 * so that we can easily arrange it */ | ||||||
|  | 		old_state = l->layer_surface->current; | ||||||
|  | 		l->layer_surface->current = l->layer_surface->pending; | ||||||
|  | 		arrangelayers(l->mon); | ||||||
|  | 		l->layer_surface->current = old_state; | ||||||
|  | 		return; | ||||||
|  | 	} | ||||||
| 
 | 
 | ||||||
| 	if (layer_surface->current.committed == 0 && l->mapped == layer_surface->surface->mapped) | 	if (layer_surface->current.committed == 0 && l->mapped == layer_surface->surface->mapped) | ||||||
| 		return; | 		return; | ||||||
| @ -896,6 +915,12 @@ commitnotify(struct wl_listener *listener, void *data) | |||||||
| 		wlr_surface_set_preferred_buffer_scale(client_surface(c), (int)ceilf(c->mon->wlr_output->scale)); | 		wlr_surface_set_preferred_buffer_scale(client_surface(c), (int)ceilf(c->mon->wlr_output->scale)); | ||||||
| 		wlr_fractional_scale_v1_notify_scale(client_surface(c), c->mon->wlr_output->scale); | 		wlr_fractional_scale_v1_notify_scale(client_surface(c), c->mon->wlr_output->scale); | ||||||
| 		setmon(c, NULL, 0); /* Make sure to reapply rules in mapnotify() */ | 		setmon(c, NULL, 0); /* Make sure to reapply rules in mapnotify() */ | ||||||
|  | 
 | ||||||
|  | 		wlr_xdg_toplevel_set_wm_capabilities(c->surface.xdg->toplevel, WLR_XDG_TOPLEVEL_WM_CAPABILITIES_FULLSCREEN); | ||||||
|  | 		wlr_xdg_toplevel_set_size(c->surface.xdg->toplevel, 0, 0); | ||||||
|  | 		if (c->decoration) | ||||||
|  | 			requestdecorationmode(&c->set_decoration_mode, c->decoration); | ||||||
|  | 		return; | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	if (client_surface(c)->mapped && c->mon) | 	if (client_surface(c)->mapped && c->mon) | ||||||
| @ -906,6 +931,33 @@ commitnotify(struct wl_listener *listener, void *data) | |||||||
| 		c->resize = 0; | 		c->resize = 0; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | void | ||||||
|  | commitpopup(struct wl_listener *listener, void *data) | ||||||
|  | { | ||||||
|  | 	struct wlr_surface *surface = data; | ||||||
|  | 	struct wlr_xdg_popup *popup = wlr_xdg_popup_try_from_wlr_surface(surface); | ||||||
|  | 	LayerSurface *l = NULL; | ||||||
|  | 	Client *c = NULL; | ||||||
|  | 	struct wlr_box box; | ||||||
|  | 	int type = -1; | ||||||
|  | 
 | ||||||
|  | 	if (!popup->base->initial_commit) | ||||||
|  | 		return; | ||||||
|  | 
 | ||||||
|  | 	type = toplevel_from_wlr_surface(popup->base->surface, &c, &l); | ||||||
|  | 	if (!popup->parent || type < 0) | ||||||
|  | 		return; | ||||||
|  | 	popup->base->surface->data = wlr_scene_xdg_surface_create( | ||||||
|  | 			popup->parent->data, popup->base); | ||||||
|  | 	if ((l && !l->mon) || (c && !c->mon)) | ||||||
|  | 		return; | ||||||
|  | 	box = type == LayerShell ? l->mon->m : c->mon->w; | ||||||
|  | 	box.x -= (type == LayerShell ? l->geom.x : c->geom.x); | ||||||
|  | 	box.y -= (type == LayerShell ? l->geom.y : c->geom.y); | ||||||
|  | 	wlr_xdg_popup_unconstrain_from_box(popup, &box); | ||||||
|  | 	wl_list_remove(&listener->link); | ||||||
|  | } | ||||||
|  | 
 | ||||||
| void | void | ||||||
| createdecoration(struct wl_listener *listener, void *data) | createdecoration(struct wl_listener *listener, void *data) | ||||||
| { | { | ||||||
| @ -964,8 +1016,7 @@ createkeyboardgroup(void) | |||||||
| 	LISTEN(&group->wlr_group->keyboard.events.key, &group->key, keypress); | 	LISTEN(&group->wlr_group->keyboard.events.key, &group->key, keypress); | ||||||
| 	LISTEN(&group->wlr_group->keyboard.events.modifiers, &group->modifiers, keypressmod); | 	LISTEN(&group->wlr_group->keyboard.events.modifiers, &group->modifiers, keypressmod); | ||||||
| 
 | 
 | ||||||
| 	group->key_repeat_source = wl_event_loop_add_timer( | 	group->key_repeat_source = wl_event_loop_add_timer(event_loop, keyrepeat, group); | ||||||
| 			wl_display_get_event_loop(dpy), keyrepeat, group); |  | ||||||
| 
 | 
 | ||||||
| 	/* A seat can only have one keyboard, but this is a limitation of the
 | 	/* A seat can only have one keyboard, but this is a limitation of the
 | ||||||
| 	 * Wayland protocol - not wlroots. We assign all connected keyboards to the | 	 * Wayland protocol - not wlroots. We assign all connected keyboards to the | ||||||
| @ -983,7 +1034,6 @@ createlayersurface(struct wl_listener *listener, void *data) | |||||||
| 	LayerSurface *l; | 	LayerSurface *l; | ||||||
| 	struct wlr_surface *surface = layer_surface->surface; | 	struct wlr_surface *surface = layer_surface->surface; | ||||||
| 	struct wlr_scene_tree *scene_layer = layers[layermap[layer_surface->pending.layer]]; | 	struct wlr_scene_tree *scene_layer = layers[layermap[layer_surface->pending.layer]]; | ||||||
| 	struct wlr_layer_surface_v1_state old_state; |  | ||||||
| 
 | 
 | ||||||
| 	if (!layer_surface->output | 	if (!layer_surface->output | ||||||
| 			&& !(layer_surface->output = selmon ? selmon->wlr_output : NULL)) { | 			&& !(layer_surface->output = selmon ? selmon->wlr_output : NULL)) { | ||||||
| @ -1009,15 +1059,6 @@ createlayersurface(struct wl_listener *listener, void *data) | |||||||
| 	wlr_surface_send_enter(surface, layer_surface->output); | 	wlr_surface_send_enter(surface, layer_surface->output); | ||||||
| 	wlr_fractional_scale_v1_notify_scale(surface, l->mon->wlr_output->scale); | 	wlr_fractional_scale_v1_notify_scale(surface, l->mon->wlr_output->scale); | ||||||
| 	wlr_surface_set_preferred_buffer_scale(surface, (int32_t)ceilf(l->mon->wlr_output->scale)); | 	wlr_surface_set_preferred_buffer_scale(surface, (int32_t)ceilf(l->mon->wlr_output->scale)); | ||||||
| 
 |  | ||||||
| 	/* Temporarily set the layer's current state to pending
 |  | ||||||
| 	 * so that we can easily arrange it |  | ||||||
| 	 */ |  | ||||||
| 	old_state = layer_surface->current; |  | ||||||
| 	layer_surface->current = layer_surface->pending; |  | ||||||
| 	l->mapped = 1; |  | ||||||
| 	arrangelayers(l->mon); |  | ||||||
| 	layer_surface->current = old_state; |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void | void | ||||||
| @ -1131,49 +1172,22 @@ createmon(struct wl_listener *listener, void *data) | |||||||
| void | void | ||||||
| createnotify(struct wl_listener *listener, void *data) | createnotify(struct wl_listener *listener, void *data) | ||||||
| { | { | ||||||
| 	/* This event is raised when wlr_xdg_shell receives a new xdg surface from a
 | 	/* This event is raised when a client creates a new toplevel (application window). */ | ||||||
| 	 * client, either a toplevel (application window) or popup, | 	struct wlr_xdg_toplevel *toplevel = data; | ||||||
| 	 * or when wlr_layer_shell receives a new popup from a layer. |  | ||||||
| 	 * If you want to do something tricky with popups you should check if |  | ||||||
| 	 * its parent is wlr_xdg_shell or wlr_layer_shell */ |  | ||||||
| 	struct wlr_xdg_surface *xdg_surface = data; |  | ||||||
| 	Client *c = NULL; | 	Client *c = NULL; | ||||||
| 	LayerSurface *l = NULL; |  | ||||||
| 
 |  | ||||||
| 	if (xdg_surface->role == WLR_XDG_SURFACE_ROLE_POPUP) { |  | ||||||
| 		struct wlr_xdg_popup *popup = xdg_surface->popup; |  | ||||||
| 		struct wlr_box box; |  | ||||||
| 		if (toplevel_from_wlr_surface(popup->base->surface, &c, &l) < 0) |  | ||||||
| 			return; |  | ||||||
| 		popup->base->surface->data = wlr_scene_xdg_surface_create( |  | ||||||
| 				popup->parent->data, popup->base); |  | ||||||
| 		if ((l && !l->mon) || (c && !c->mon)) |  | ||||||
| 			return; |  | ||||||
| 		box = l ? l->mon->m : c->mon->w; |  | ||||||
| 		box.x -= (l ? l->geom.x : c->geom.x); |  | ||||||
| 		box.y -= (l ? l->geom.y : c->geom.y); |  | ||||||
| 		wlr_xdg_popup_unconstrain_from_box(popup, &box); |  | ||||||
| 		return; |  | ||||||
| 	} else if (xdg_surface->role == WLR_XDG_SURFACE_ROLE_NONE) |  | ||||||
| 		return; |  | ||||||
| 
 | 
 | ||||||
| 	/* Allocate a Client for this surface */ | 	/* Allocate a Client for this surface */ | ||||||
| 	c = xdg_surface->data = ecalloc(1, sizeof(*c)); | 	c = toplevel->base->data = ecalloc(1, sizeof(*c)); | ||||||
| 	c->surface.xdg = xdg_surface; | 	c->surface.xdg = toplevel->base; | ||||||
| 	c->bw = borderpx; | 	c->bw = borderpx; | ||||||
| 
 | 
 | ||||||
| 	wlr_xdg_toplevel_set_wm_capabilities(xdg_surface->toplevel, | 	LISTEN(&toplevel->base->surface->events.commit, &c->commit, commitnotify); | ||||||
| 			WLR_XDG_TOPLEVEL_WM_CAPABILITIES_FULLSCREEN); | 	LISTEN(&toplevel->base->surface->events.map, &c->map, mapnotify); | ||||||
| 
 | 	LISTEN(&toplevel->base->surface->events.unmap, &c->unmap, unmapnotify); | ||||||
| 	LISTEN(&xdg_surface->events.destroy, &c->destroy, destroynotify); | 	LISTEN(&toplevel->events.destroy, &c->destroy, destroynotify); | ||||||
| 	LISTEN(&xdg_surface->surface->events.commit, &c->commit, commitnotify); | 	LISTEN(&toplevel->events.request_fullscreen, &c->fullscreen, fullscreennotify); | ||||||
| 	LISTEN(&xdg_surface->surface->events.map, &c->map, mapnotify); | 	LISTEN(&toplevel->events.request_maximize, &c->maximize, maximizenotify); | ||||||
| 	LISTEN(&xdg_surface->surface->events.unmap, &c->unmap, unmapnotify); | 	LISTEN(&toplevel->events.set_title, &c->set_title, updatetitle); | ||||||
| 	LISTEN(&xdg_surface->toplevel->events.request_fullscreen, &c->fullscreen, |  | ||||||
| 			fullscreennotify); |  | ||||||
| 	LISTEN(&xdg_surface->toplevel->events.request_maximize, &c->maximize, |  | ||||||
| 			maximizenotify); |  | ||||||
| 	LISTEN(&xdg_surface->toplevel->events.set_title, &c->set_title, updatetitle); |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void | void | ||||||
| @ -1229,6 +1243,15 @@ createpointerconstraint(struct wl_listener *listener, void *data) | |||||||
| 			&pointer_constraint->destroy, destroypointerconstraint); | 			&pointer_constraint->destroy, destroypointerconstraint); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | void | ||||||
|  | createpopup(struct wl_listener *listener, void *data) | ||||||
|  | { | ||||||
|  | 	/* This event is raised when a client (either xdg-shell or layer-shell)
 | ||||||
|  | 	 * creates a new popup. */ | ||||||
|  | 	struct wlr_xdg_popup *popup = data; | ||||||
|  | 	LISTEN_STATIC(&popup->base->surface->events.commit, commitpopup); | ||||||
|  | } | ||||||
|  | 
 | ||||||
| void | void | ||||||
| cursorconstrain(struct wlr_pointer_constraint_v1 *constraint) | cursorconstrain(struct wlr_pointer_constraint_v1 *constraint) | ||||||
| { | { | ||||||
| @ -1261,8 +1284,7 @@ cursorwarptohint(void) | |||||||
| 	double sy = active_constraint->current.cursor_hint.y; | 	double sy = active_constraint->current.cursor_hint.y; | ||||||
| 
 | 
 | ||||||
| 	toplevel_from_wlr_surface(active_constraint->surface, &c, NULL); | 	toplevel_from_wlr_surface(active_constraint->surface, &c, NULL); | ||||||
| 	/* TODO: wlroots 0.18: https://gitlab.freedesktop.org/wlroots/wlroots/-/merge_requests/4478 */ | 	if (c && active_constraint->current.cursor_hint.enabled) { | ||||||
| 	if (c && (active_constraint->current.committed & WLR_POINTER_CONSTRAINT_V1_STATE_CURSOR_HINT )) { |  | ||||||
| 		wlr_cursor_warp(cursor, NULL, sx + c->geom.x + c->bw, sy + c->geom.y + c->bw); | 		wlr_cursor_warp(cursor, NULL, sx + c->geom.x + c->bw, sy + c->geom.y + c->bw); | ||||||
| 		wlr_seat_pointer_warp(active_constraint->seat, sx, sy); | 		wlr_seat_pointer_warp(active_constraint->seat, sx, sy); | ||||||
| 	} | 	} | ||||||
| @ -1272,6 +1294,7 @@ void | |||||||
| destroydecoration(struct wl_listener *listener, void *data) | destroydecoration(struct wl_listener *listener, void *data) | ||||||
| { | { | ||||||
| 	Client *c = wl_container_of(listener, c, destroy_decoration); | 	Client *c = wl_container_of(listener, c, destroy_decoration); | ||||||
|  | 	c->decoration = NULL; | ||||||
| 
 | 
 | ||||||
| 	wl_list_remove(&c->destroy_decoration.link); | 	wl_list_remove(&c->destroy_decoration.link); | ||||||
| 	wl_list_remove(&c->set_decoration_mode.link); | 	wl_list_remove(&c->set_decoration_mode.link); | ||||||
| @ -1648,6 +1671,30 @@ fullscreennotify(struct wl_listener *listener, void *data) | |||||||
| 	setfullscreen(c, client_wants_fullscreen(c)); | 	setfullscreen(c, client_wants_fullscreen(c)); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | void | ||||||
|  | gpureset(struct wl_listener *listener, void *data) | ||||||
|  | { | ||||||
|  | 	struct wlr_renderer *old_drw = drw; | ||||||
|  | 	struct wlr_allocator *old_alloc = alloc; | ||||||
|  | 	struct Monitor *m; | ||||||
|  | 	if (!(drw = wlr_renderer_autocreate(backend))) | ||||||
|  | 		die("couldn't recreate renderer"); | ||||||
|  | 
 | ||||||
|  | 	if (!(alloc = wlr_allocator_autocreate(backend, drw))) | ||||||
|  | 		die("couldn't recreate allocator"); | ||||||
|  | 
 | ||||||
|  | 	LISTEN_STATIC(&drw->events.lost, gpureset); | ||||||
|  | 
 | ||||||
|  | 	wlr_compositor_set_renderer(compositor, drw); | ||||||
|  | 
 | ||||||
|  | 	wl_list_for_each(m, &mons, link) { | ||||||
|  | 		wlr_output_init_render(m->wlr_output, alloc, drw); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	wlr_allocator_destroy(old_alloc); | ||||||
|  | 	wlr_renderer_destroy(old_drw); | ||||||
|  | } | ||||||
|  | 
 | ||||||
| void | void | ||||||
| handlesig(int signo) | handlesig(int signo) | ||||||
| { | { | ||||||
| @ -1914,7 +1961,8 @@ maximizenotify(struct wl_listener *listener, void *data) | |||||||
| 	 * protocol version | 	 * protocol version | ||||||
| 	 * wlr_xdg_surface_schedule_configure() is used to send an empty reply. */ | 	 * wlr_xdg_surface_schedule_configure() is used to send an empty reply. */ | ||||||
| 	Client *c = wl_container_of(listener, c, maximize); | 	Client *c = wl_container_of(listener, c, maximize); | ||||||
| 	if (wl_resource_get_version(c->surface.xdg->toplevel->resource) | 	if (c->surface.xdg->initialized | ||||||
|  | 			&& wl_resource_get_version(c->surface.xdg->toplevel->resource) | ||||||
| 					< XDG_TOPLEVEL_WM_CAPABILITIES_SINCE_VERSION) | 					< XDG_TOPLEVEL_WM_CAPABILITIES_SINCE_VERSION) | ||||||
| 		wlr_xdg_surface_schedule_configure(c->surface.xdg); | 		wlr_xdg_surface_schedule_configure(c->surface.xdg); | ||||||
| } | } | ||||||
| @ -2265,6 +2313,7 @@ void | |||||||
| requestdecorationmode(struct wl_listener *listener, void *data) | requestdecorationmode(struct wl_listener *listener, void *data) | ||||||
| { | { | ||||||
| 	Client *c = wl_container_of(listener, c, set_decoration_mode); | 	Client *c = wl_container_of(listener, c, set_decoration_mode); | ||||||
|  | 	if (c->surface.xdg->initialized) | ||||||
| 		wlr_xdg_toplevel_decoration_v1_set_mode(c->decoration, | 		wlr_xdg_toplevel_decoration_v1_set_mode(c->decoration, | ||||||
| 				WLR_XDG_TOPLEVEL_DECORATION_V1_MODE_SERVER_SIDE); | 				WLR_XDG_TOPLEVEL_DECORATION_V1_MODE_SERVER_SIDE); | ||||||
| } | } | ||||||
| @ -2546,12 +2595,13 @@ setup(void) | |||||||
| 	/* The Wayland display is managed by libwayland. It handles accepting
 | 	/* The Wayland display is managed by libwayland. It handles accepting
 | ||||||
| 	 * clients from the Unix socket, manging Wayland globals, and so on. */ | 	 * clients from the Unix socket, manging Wayland globals, and so on. */ | ||||||
| 	dpy = wl_display_create(); | 	dpy = wl_display_create(); | ||||||
|  | 	event_loop = wl_display_get_event_loop(dpy); | ||||||
| 
 | 
 | ||||||
| 	/* The backend is a wlroots feature which abstracts the underlying input and
 | 	/* The backend is a wlroots feature which abstracts the underlying input and
 | ||||||
| 	 * output hardware. The autocreate option will choose the most suitable | 	 * output hardware. The autocreate option will choose the most suitable | ||||||
| 	 * backend based on the current environment, such as opening an X11 window | 	 * backend based on the current environment, such as opening an X11 window | ||||||
| 	 * if an X11 server is running. */ | 	 * if an X11 server is running. */ | ||||||
| 	if (!(backend = wlr_backend_autocreate(dpy, &session))) | 	if (!(backend = wlr_backend_autocreate(event_loop, &session))) | ||||||
| 		die("couldn't create backend"); | 		die("couldn't create backend"); | ||||||
| 
 | 
 | ||||||
| 	/* Initialize the scene graph used to lay out windows */ | 	/* Initialize the scene graph used to lay out windows */ | ||||||
| @ -2568,6 +2618,7 @@ setup(void) | |||||||
| 	 * supports for shared memory, this configures that for clients. */ | 	 * supports for shared memory, this configures that for clients. */ | ||||||
| 	if (!(drw = wlr_renderer_autocreate(backend))) | 	if (!(drw = wlr_renderer_autocreate(backend))) | ||||||
| 		die("couldn't create renderer"); | 		die("couldn't create renderer"); | ||||||
|  | 	LISTEN_STATIC(&drw->events.lost, gpureset); | ||||||
| 
 | 
 | ||||||
| 	/* Create shm, drm and linux_dmabuf interfaces by ourselves.
 | 	/* Create shm, drm and linux_dmabuf interfaces by ourselves.
 | ||||||
| 	 * The simplest way is call: | 	 * The simplest way is call: | ||||||
| @ -2576,10 +2627,10 @@ setup(void) | |||||||
| 	 * with wlr_scene. */ | 	 * with wlr_scene. */ | ||||||
| 	wlr_renderer_init_wl_shm(drw, dpy); | 	wlr_renderer_init_wl_shm(drw, dpy); | ||||||
| 
 | 
 | ||||||
| 	if (wlr_renderer_get_dmabuf_texture_formats(drw)) { | 	if (wlr_renderer_get_texture_formats(drw, WLR_BUFFER_CAP_DMABUF)) { | ||||||
| 		wlr_drm_create(dpy, drw); | 		wlr_drm_create(dpy, drw); | ||||||
| 		wlr_scene_set_linux_dmabuf_v1(scene, | 		wlr_scene_set_linux_dmabuf_v1(scene, | ||||||
| 				wlr_linux_dmabuf_v1_create_with_renderer(dpy, 4, drw)); | 				wlr_linux_dmabuf_v1_create_with_renderer(dpy, 5, drw)); | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	/* Autocreates an allocator for us.
 | 	/* Autocreates an allocator for us.
 | ||||||
| @ -2605,6 +2656,8 @@ setup(void) | |||||||
| 	wlr_viewporter_create(dpy); | 	wlr_viewporter_create(dpy); | ||||||
| 	wlr_single_pixel_buffer_manager_v1_create(dpy); | 	wlr_single_pixel_buffer_manager_v1_create(dpy); | ||||||
| 	wlr_fractional_scale_manager_v1_create(dpy, 1); | 	wlr_fractional_scale_manager_v1_create(dpy, 1); | ||||||
|  | 	wlr_presentation_create(dpy, backend); | ||||||
|  | 	wlr_alpha_modifier_v1_create(dpy); | ||||||
| 
 | 
 | ||||||
| 	/* Initializes the interface used to implement urgency hints */ | 	/* Initializes the interface used to implement urgency hints */ | ||||||
| 	activation = wlr_xdg_activation_v1_create(dpy); | 	activation = wlr_xdg_activation_v1_create(dpy); | ||||||
| @ -2618,7 +2671,7 @@ setup(void) | |||||||
| 
 | 
 | ||||||
| 	/* Creates an output layout, which a wlroots utility for working with an
 | 	/* Creates an output layout, which a wlroots utility for working with an
 | ||||||
| 	 * arrangement of screens in a physical layout. */ | 	 * arrangement of screens in a physical layout. */ | ||||||
| 	output_layout = wlr_output_layout_create(); | 	output_layout = wlr_output_layout_create(dpy); | ||||||
| 	LISTEN_STATIC(&output_layout->events.change, updatemons); | 	LISTEN_STATIC(&output_layout->events.change, updatemons); | ||||||
| 	wlr_xdg_output_manager_v1_create(dpy, output_layout); | 	wlr_xdg_output_manager_v1_create(dpy, output_layout); | ||||||
| 
 | 
 | ||||||
| @ -2637,7 +2690,8 @@ setup(void) | |||||||
| 	wl_list_init(&fstack); | 	wl_list_init(&fstack); | ||||||
| 
 | 
 | ||||||
| 	xdg_shell = wlr_xdg_shell_create(dpy, 6); | 	xdg_shell = wlr_xdg_shell_create(dpy, 6); | ||||||
| 	LISTEN_STATIC(&xdg_shell->events.new_surface, createnotify); | 	LISTEN_STATIC(&xdg_shell->events.new_toplevel, createnotify); | ||||||
|  | 	LISTEN_STATIC(&xdg_shell->events.new_popup, createpopup); | ||||||
| 
 | 
 | ||||||
| 	layer_shell = wlr_layer_shell_v1_create(dpy, 3); | 	layer_shell = wlr_layer_shell_v1_create(dpy, 3); | ||||||
| 	LISTEN_STATIC(&layer_shell->events.new_surface, createlayersurface); | 	LISTEN_STATIC(&layer_shell->events.new_surface, createlayersurface); | ||||||
| @ -2725,13 +2779,6 @@ setup(void) | |||||||
| 	LISTEN_STATIC(&output_mgr->events.apply, outputmgrapply); | 	LISTEN_STATIC(&output_mgr->events.apply, outputmgrapply); | ||||||
| 	LISTEN_STATIC(&output_mgr->events.test, outputmgrtest); | 	LISTEN_STATIC(&output_mgr->events.test, outputmgrtest); | ||||||
| 
 | 
 | ||||||
| 	wlr_scene_set_presentation(scene, wlr_presentation_create(dpy, backend)); |  | ||||||
| 
 |  | ||||||
| 	drwl_init(); |  | ||||||
| 
 |  | ||||||
| 	status_event_source = wl_event_loop_add_fd(wl_display_get_event_loop(dpy), |  | ||||||
| 		STDIN_FILENO, WL_EVENT_READABLE, status_in, NULL); |  | ||||||
| 
 |  | ||||||
| 	/* Make sure XWayland clients don't connect to the parent X server,
 | 	/* Make sure XWayland clients don't connect to the parent X server,
 | ||||||
| 	 * e.g when running in the x11 backend or the wayland backend and the | 	 * e.g when running in the x11 backend or the wayland backend and the | ||||||
| 	 * compositor has Xwayland support */ | 	 * compositor has Xwayland support */ | ||||||
|  | |||||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user
	 Wilkenfeld
						Wilkenfeld