mirror of
				https://codeberg.org/dwl/dwl-patches.git
				synced 2025-10-27 10:14:16 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			462 lines
		
	
	
		
			16 KiB
		
	
	
	
		
			Diff
		
	
	
	
	
	
			
		
		
	
	
			462 lines
		
	
	
		
			16 KiB
		
	
	
	
		
			Diff
		
	
	
	
	
	
| From b624206781513cdff1b9609fc5ac4b848094e1b4 Mon Sep 17 00:00:00 2001
 | |
| From: Gavin M <github@gavinm.us>
 | |
| Date: Fri, 15 Mar 2024 16:37:23 -0500
 | |
| Subject: [PATCH] Tabbed patch
 | |
| 
 | |
| ---
 | |
|  Makefile     |   2 +-
 | |
|  config.def.h |  18 +++-
 | |
|  dwl.c        | 276 +++++++++++++++++++++++++++++++++++++++++++++++++--
 | |
|  3 files changed, 281 insertions(+), 15 deletions(-)
 | |
| 
 | |
| diff --git a/Makefile b/Makefile
 | |
| index a67fdd3..182eb87 100644
 | |
| --- a/Makefile
 | |
| +++ b/Makefile
 | |
| @@ -9,7 +9,7 @@ DWLDEVCFLAGS = -g -pedantic -Wall -Wextra -Wdeclaration-after-statement -Wno-unu
 | |
|  	-Werror=strict-prototypes -Werror=implicit -Werror=return-type -Werror=incompatible-pointer-types -Wfloat-conversion
 | |
|  
 | |
|  # CFLAGS / LDFLAGS
 | |
| -PKGS      = wlroots wayland-server xkbcommon libinput $(XLIBS)
 | |
| +PKGS      = wlroots wayland-server xkbcommon libinput cairo pangocairo $(XLIBS)
 | |
|  DWLCFLAGS = `$(PKG_CONFIG) --cflags $(PKGS)` $(DWLCPPFLAGS) $(DWLDEVCFLAGS) $(CFLAGS)
 | |
|  LDLIBS    = `$(PKG_CONFIG) --libs $(PKGS)` $(LIBS)
 | |
|  
 | |
| diff --git a/config.def.h b/config.def.h
 | |
| index 9009517..1ca270f 100644
 | |
| --- a/config.def.h
 | |
| +++ b/config.def.h
 | |
| @@ -7,6 +7,16 @@
 | |
|  static const int sloppyfocus               = 1;  /* focus follows mouse */
 | |
|  static const int bypass_surface_visibility = 0;  /* 1 means idle inhibitors will disable idle tracking even if it's surface isn't visible  */
 | |
|  static const unsigned int borderpx         = 1;  /* border pixel of windows */
 | |
| +static const double title_border_width     = 0.75;
 | |
| +static const unsigned int title_padding    = 11;
 | |
| +static const int title_top                 = 0;
 | |
| +static const LayoutType floating_title_type = LAYOUT_TYPE_LABEL;
 | |
| +static const char title_font[]             = "Dejavu Sans Mono 10.5";
 | |
| +static const float title_font_color[]      = COLOR(0xffffffff);
 | |
| +static const float title_focus_bg[]        = COLOR(0x3b3b3bff);
 | |
| +static const float title_root_bg[]         = COLOR(0x131313ff);
 | |
| +static const float title_urgent_bg[]       = COLOR(0x00ff00ff);
 | |
| +static const float title_border_color[]    = COLOR(0x3b3b3bff);
 | |
|  static const float rootcolor[]             = COLOR(0x222222ff);
 | |
|  static const float bordercolor[]           = COLOR(0x444444ff);
 | |
|  static const float focuscolor[]            = COLOR(0x005577ff);
 | |
| @@ -30,10 +40,10 @@ static const Rule rules[] = {
 | |
|  
 | |
|  /* layout(s) */
 | |
|  static const Layout layouts[] = {
 | |
| -	/* symbol     arrange function */
 | |
| -	{ "[]=",      tile },
 | |
| -	{ "><>",      NULL },    /* no layout function means floating behavior */
 | |
| -	{ "[M]",      monocle },
 | |
| +	/* symbol    type               render_only_top          arrange function */
 | |
| +	{ "[]=",     LAYOUT_TYPE_NONE,  0,                      tile },
 | |
| +	{ "><>",     LAYOUT_TYPE_LABEL, 0,                      NULL },    /* no layout function means floating behavior */
 | |
| +	{ "[M]",     LAYOUT_TYPE_TABS_ONLY_MULTIPLE_CLIENTS, 1, monocle },
 | |
|  };
 | |
|  
 | |
|  /* monitors */
 | |
| diff --git a/dwl.c b/dwl.c
 | |
| index 5867b0c..e613d17 100644
 | |
| --- a/dwl.c
 | |
| +++ b/dwl.c
 | |
| @@ -2,6 +2,11 @@
 | |
|   * See LICENSE file for copyright and license details.
 | |
|   */
 | |
|  #include <getopt.h>
 | |
| +#include <cairo/cairo.h>
 | |
| +#include <pango/pangocairo.h>
 | |
| +#include <pango/pango-font.h>
 | |
| +#include <pango/pango-layout.h>
 | |
| +#include <libdrm/drm_fourcc.h>
 | |
|  #include <libinput.h>
 | |
|  #include <linux/input-event-codes.h>
 | |
|  #include <signal.h>
 | |
| @@ -13,8 +18,10 @@
 | |
|  #include <wayland-server-core.h>
 | |
|  #include <wlr/backend.h>
 | |
|  #include <wlr/backend/libinput.h>
 | |
| +#include <wlr/interfaces/wlr_buffer.h>
 | |
|  #include <wlr/render/allocator.h>
 | |
|  #include <wlr/render/wlr_renderer.h>
 | |
| +#include <wlr/types/wlr_buffer.h>
 | |
|  #include <wlr/types/wlr_compositor.h>
 | |
|  #include <wlr/types/wlr_cursor.h>
 | |
|  #include <wlr/types/wlr_cursor_shape_v1.h>
 | |
| @@ -110,6 +117,7 @@ typedef struct {
 | |
|  	struct wlr_scene_tree *scene;
 | |
|  	struct wlr_scene_rect *border[4]; /* top, bottom, left, right */
 | |
|  	struct wlr_scene_tree *scene_surface;
 | |
| +	struct wlr_scene_buffer *titlebar;
 | |
|  	struct wl_list link;
 | |
|  	struct wl_list flink;
 | |
|  	union {
 | |
| @@ -137,7 +145,7 @@ typedef struct {
 | |
|  #endif
 | |
|  	unsigned int bw;
 | |
|  	uint32_t tags;
 | |
| -	int isfloating, isurgent, isfullscreen;
 | |
| +	int isfloating, isurgent, isfullscreen, titleisinit, istabbed;
 | |
|  	uint32_t resize; /* configure serial of a pending resize */
 | |
|  } Client;
 | |
|  
 | |
| @@ -179,8 +187,17 @@ typedef struct {
 | |
|  	struct wl_listener surface_commit;
 | |
|  } LayerSurface;
 | |
|  
 | |
| +typedef enum {
 | |
| +	LAYOUT_TYPE_NONE,
 | |
| +	LAYOUT_TYPE_LABEL,
 | |
| +	LAYOUT_TYPE_TABS_ONLY_MULTIPLE_CLIENTS,
 | |
| +	LAYOUT_TYPE_TABS_ALWAYS
 | |
| +} LayoutType;
 | |
| +
 | |
|  typedef struct {
 | |
|  	const char *symbol;
 | |
| +	LayoutType type;
 | |
| +	int render_top_only;
 | |
|  	void (*arrange)(Monitor *);
 | |
|  } Layout;
 | |
|  
 | |
| @@ -282,6 +299,7 @@ static void focusclient(Client *c, int lift);
 | |
|  static void focusmon(const Arg *arg);
 | |
|  static void focusstack(const Arg *arg);
 | |
|  static Client *focustop(Monitor *m);
 | |
| +static Client *focustop_onlytiled(Monitor *m, int onlytiled);
 | |
|  static void fullscreennotify(struct wl_listener *listener, void *data);
 | |
|  static void handlesig(int signo);
 | |
|  static void incnmaster(const Arg *arg);
 | |
| @@ -309,6 +327,7 @@ static void pointerfocus(Client *c, struct wlr_surface *surface,
 | |
|  static void printstatus(void);
 | |
|  static void quit(const Arg *arg);
 | |
|  static void rendermon(struct wl_listener *listener, void *data);
 | |
| +static void rendertitlebar(Client *client);
 | |
|  static void requestdecorationmode(struct wl_listener *listener, void *data);
 | |
|  static void requeststartdrag(struct wl_listener *listener, void *data);
 | |
|  static void requestmonstate(struct wl_listener *listener, void *data);
 | |
| @@ -349,6 +368,7 @@ static void xytonode(double x, double y, struct wlr_surface **psurface,
 | |
|  static void zoom(const Arg *arg);
 | |
|  
 | |
|  /* variables */
 | |
| +static int title_height;
 | |
|  static const char broken[] = "broken";
 | |
|  static pid_t child_pid = -1;
 | |
|  static int locked;
 | |
| @@ -973,6 +993,7 @@ createnotify(struct wl_listener *listener, void *data)
 | |
|  	c = xdg_surface->data = ecalloc(1, sizeof(*c));
 | |
|  	c->surface.xdg = xdg_surface;
 | |
|  	c->bw = borderpx;
 | |
| +	c->titleisinit = c->istabbed = 0;
 | |
|  
 | |
|  	wlr_xdg_toplevel_set_wm_capabilities(xdg_surface->toplevel,
 | |
|  			WLR_XDG_TOPLEVEL_WM_CAPABILITIES_FULLSCREEN);
 | |
| @@ -1360,6 +1381,22 @@ focustop(Monitor *m)
 | |
|  	return NULL;
 | |
|  }
 | |
|  
 | |
| +Client *
 | |
| +focustop_onlytiled(Monitor *m, int onlytiled)
 | |
| +{
 | |
| +	Client *c;
 | |
| +	if (!m)
 | |
| +		return NULL;
 | |
| +	wl_list_for_each(c, &fstack, flink) {
 | |
| +		if (VISIBLEON(c, m)) {
 | |
| +			if ((onlytiled == 1 && c->isfloating) || (onlytiled == 2 && (!c->isfloating || !m->lt[m->sellt]->arrange)))
 | |
| +				continue;
 | |
| +			return c;
 | |
| +		}
 | |
| +	}
 | |
| +	return NULL;
 | |
| +}
 | |
| +
 | |
|  void
 | |
|  fullscreennotify(struct wl_listener *listener, void *data)
 | |
|  {
 | |
| @@ -2003,6 +2040,195 @@ skip:
 | |
|  	wlr_output_state_finish(&pending);
 | |
|  }
 | |
|  
 | |
| +struct text_buffer {
 | |
| +	struct wlr_buffer base;
 | |
| +	void *data;
 | |
| +	uint32_t format;
 | |
| +	size_t stride;
 | |
| +};
 | |
| +
 | |
| +static void text_buffer_destroy(struct wlr_buffer *wlr_buffer) {
 | |
| +	struct text_buffer *buffer = wl_container_of(wlr_buffer, buffer, base);
 | |
| +	free(buffer->data);
 | |
| +	free(buffer);
 | |
| +}
 | |
| +
 | |
| +static bool text_buffer_begin_data_ptr_access(struct wlr_buffer *wlr_buffer,
 | |
| +		uint32_t flags, void **data, uint32_t *format, size_t *stride) {
 | |
| +	struct text_buffer *buffer = wl_container_of(wlr_buffer, buffer, base);
 | |
| +	if(data != NULL) {
 | |
| +		*data = (void *)buffer->data;
 | |
| +	}
 | |
| +	if(format != NULL) {
 | |
| +		*format = buffer->format;
 | |
| +	}
 | |
| +	if(stride != NULL) {
 | |
| +		*stride = buffer->stride;
 | |
| +	}
 | |
| +	return true;
 | |
| +}
 | |
| +
 | |
| +static void text_buffer_end_data_ptr_access(struct wlr_buffer *wlr_buffer) {
 | |
| +	// This space is intentionally left blank
 | |
| +}
 | |
| +
 | |
| +static const struct wlr_buffer_impl text_buffer_impl = {
 | |
| +	.destroy = text_buffer_destroy,
 | |
| +	.begin_data_ptr_access = text_buffer_begin_data_ptr_access,
 | |
| +	.end_data_ptr_access = text_buffer_end_data_ptr_access,
 | |
| +};
 | |
| +
 | |
| +static struct text_buffer *text_buffer_create(uint32_t width, uint32_t height, uint32_t stride) {
 | |
| +	struct text_buffer *buffer = calloc(1, sizeof(*buffer));
 | |
| +	if (buffer == NULL) {
 | |
| +		return NULL;
 | |
| +	}
 | |
| +
 | |
| +	wlr_buffer_init(&buffer->base, &text_buffer_impl, width, height);
 | |
| +	buffer->format = DRM_FORMAT_ARGB8888;
 | |
| +	buffer->stride = stride;
 | |
| +
 | |
| +	buffer->data = malloc(buffer->stride * height);
 | |
| +	if (buffer->data == NULL) {
 | |
| +		free(buffer);
 | |
| +		return NULL;
 | |
| +	}
 | |
| +
 | |
| +	return buffer;
 | |
| +}
 | |
| +
 | |
| +void
 | |
| +rendertitlebar(Client *c)
 | |
| +{
 | |
| +	struct wl_list *init_destroy, *cursor_destroy;
 | |
| +	cairo_surface_t *surface;
 | |
| +	cairo_status_t status;
 | |
| +	cairo_t *cr;
 | |
| +	PangoFontDescription *desc;
 | |
| +	PangoLayout *layout;
 | |
| +	LayoutType title_type;
 | |
| +	Client *l, *sel;
 | |
| +	unsigned int len, tabsize, i;
 | |
| +	const char *title;
 | |
| +	const float *color;
 | |
| +	unsigned char *data;
 | |
| +	int stride;
 | |
| +	struct text_buffer *text_buffer;
 | |
| +	void *data_ptr;
 | |
| +
 | |
| +	if (!c || !c->scene || !c->mon || !selmon || (!VISIBLEON(c, selmon) && c->mon == selmon))
 | |
| +		return;
 | |
| +
 | |
| +	if (c->titleisinit) {
 | |
| +		init_destroy = cursor_destroy = &c->titlebar->node.events.destroy.listener_list;
 | |
| +		do {
 | |
| +		  cursor_destroy = cursor_destroy->next;
 | |
| +		} while (cursor_destroy && cursor_destroy != init_destroy);
 | |
| +		if (!cursor_destroy) {
 | |
| +		  return;
 | |
| +		}
 | |
| +		wlr_scene_node_destroy(&c->titlebar->node);
 | |
| +	}
 | |
| +	c->titleisinit = c->istabbed = 0;
 | |
| +
 | |
| +	sel = focustop_onlytiled(c->mon, c->isfloating + 1);
 | |
| +
 | |
| +	if (c->isfullscreen)
 | |
| +		return;
 | |
| +	title_type = c->isfloating ? floating_title_type : c->mon->lt[c->mon->sellt]->type;
 | |
| +
 | |
| +	if (title_type == LAYOUT_TYPE_TABS_ONLY_MULTIPLE_CLIENTS || title_type == LAYOUT_TYPE_TABS_ALWAYS) {
 | |
| +		len = 0;
 | |
| +		wl_list_for_each(l, &clients, link) {
 | |
| +			if (VISIBLEON(l, c->mon) && l->isfloating == c->isfloating)
 | |
| +				len++;
 | |
| +		}
 | |
| +		if (title_type == LAYOUT_TYPE_TABS_ONLY_MULTIPLE_CLIENTS && len <= 1)
 | |
| +			return;
 | |
| +	}
 | |
| +
 | |
| +	if (c->mon->lt[c->mon->sellt]->render_top_only == 1 && !c->isfloating && c != sel) {
 | |
| +		c->istabbed = 1;
 | |
| +		return;
 | |
| +	} /*else if (c->mon->lt[c->mon->sellt]->render_top_only == 2 && c != sel) {
 | |
| +		c->istabbed = 1;
 | |
| +		return;
 | |
| +	}*/
 | |
| +
 | |
| +	if (title_type == LAYOUT_TYPE_NONE)
 | |
| +		return;
 | |
| +  
 | |
| +	surface = cairo_image_surface_create(CAIRO_FORMAT_ARGB32, c->geom.width, title_height);
 | |
| +	if ((status = cairo_surface_status(surface)) != CAIRO_STATUS_SUCCESS) {
 | |
| +		wlr_log(WLR_ERROR, "cairo_image_surface_create failed: %s",
 | |
| +			cairo_status_to_string(status));
 | |
| +		return;
 | |
| +	}
 | |
| +	cr = cairo_create(surface);
 | |
| +	desc = pango_font_description_from_string(title_font);
 | |
| +	layout = pango_cairo_create_layout(cr);
 | |
| +	pango_layout_set_font_description(layout, desc);
 | |
| +	pango_layout_set_ellipsize(layout, PANGO_ELLIPSIZE_NONE);
 | |
| +
 | |
| +	cairo_set_line_width(cr, title_border_width);
 | |
| +
 | |
| +	if (title_type == LAYOUT_TYPE_LABEL) {
 | |
| +		cairo_rectangle(cr, 0, 0, c->geom.width, title_height);
 | |
| +		cairo_set_source_rgba(cr, title_focus_bg[0], title_focus_bg[1], title_focus_bg[2], title_focus_bg[3]);
 | |
| +		cairo_fill_preserve(cr);
 | |
| +		cairo_set_source_rgba(cr, title_border_color[0], title_border_color[1], title_border_color[2], title_border_color[3]);
 | |
| +		cairo_stroke(cr);
 | |
| +		cairo_set_source_rgba(cr, title_font_color[0], title_font_color[1], title_font_color[2], title_font_color[3]);
 | |
| +		title = client_get_title(c);
 | |
| +		pango_layout_set_text(layout, title ? title : " ", (c->geom.width - title_padding) * PANGO_SCALE);
 | |
| +		cairo_move_to(cr, title_padding, 0);
 | |
| +		pango_cairo_show_layout(cr, layout);
 | |
| +	} else {
 | |
| +		tabsize = c->geom.width / len;
 | |
| +		i = 0;
 | |
| +		wl_list_for_each(l, &clients, link) {
 | |
| +			if (VISIBLEON(l, c->mon) && l->isfloating == c->isfloating) {
 | |
| +				cairo_rectangle(cr, i * tabsize, 0, (i + 1) * tabsize, title_height);
 | |
| +				color = (l == sel) ? title_focus_bg
 | |
| +					: (c->isurgent ? title_urgent_bg : title_root_bg);
 | |
| +				cairo_set_source_rgba(cr, color[0], color[1], color[2], color[3]);
 | |
| +				cairo_fill_preserve(cr);
 | |
| +				cairo_set_source_rgba(cr, title_border_color[0], title_border_color[1], title_border_color[2], title_border_color[3]);
 | |
| +				cairo_stroke(cr);
 | |
| +				cairo_set_source_rgba(cr, title_font_color[0], title_font_color[1], title_font_color[2], title_font_color[3]);
 | |
| +				title = client_get_title(l);
 | |
| +				pango_layout_set_text(layout, title ? title : " ", (tabsize - title_padding) * PANGO_SCALE);
 | |
| +				cairo_move_to(cr, (i * tabsize) + title_padding, 0);
 | |
| +				pango_cairo_show_layout(cr, layout);
 | |
| +				i++;
 | |
| +			}
 | |
| +		}
 | |
| +	}
 | |
| +  
 | |
| +	data = cairo_image_surface_get_data(surface);
 | |
| +	stride = cairo_image_surface_get_stride(surface);
 | |
| +	text_buffer = text_buffer_create(c->geom.width, title_height, stride);
 | |
| +
 | |
| +	if(!wlr_buffer_begin_data_ptr_access(&text_buffer->base,
 | |
| +      WLR_BUFFER_DATA_PTR_ACCESS_WRITE, &data_ptr, NULL, NULL)) {
 | |
| +		wlr_log(WLR_ERROR, "%s", "Failed to get pointer access to text buffer");
 | |
| +		return;
 | |
| +	}
 | |
| +	memcpy(data_ptr, data, stride * title_height);
 | |
| +	wlr_buffer_end_data_ptr_access(&text_buffer->base);
 | |
| +	cairo_surface_destroy(surface);
 | |
| +
 | |
| +	c->titlebar = wlr_scene_buffer_create(c->scene, &text_buffer->base);
 | |
| +	c->titleisinit = c->istabbed = 1;
 | |
| +
 | |
| +	wlr_scene_node_set_position(&c->titlebar->node, 0, !title_top ? c->geom.height - title_height : 0);
 | |
| +	wlr_scene_node_raise_to_top(&c->titlebar->node);
 | |
| +
 | |
| +	pango_font_description_free(desc);
 | |
| +	g_object_unref(layout);
 | |
| +	cairo_destroy(cr);
 | |
| +}
 | |
| +
 | |
|  void
 | |
|  requestdecorationmode(struct wl_listener *listener, void *data)
 | |
|  {
 | |
| @@ -2036,24 +2262,30 @@ resize(Client *c, struct wlr_box geo, int interact)
 | |
|  {
 | |
|  	struct wlr_box *bbox = interact ? &sgeom : &c->mon->w;
 | |
|  	struct wlr_box clip;
 | |
| +	unsigned int th;
 | |
| +	int draw_borders = 1;
 | |
|  	client_set_bounds(c, geo.width, geo.height);
 | |
|  	c->geom = geo;
 | |
| +	c->bw = draw_borders ? borderpx : 0;
 | |
|  	applybounds(c, bbox);
 | |
|  
 | |
| +	rendertitlebar(c);
 | |
| +	th = c->istabbed ? title_height : c->bw;
 | |
| +
 | |
|  	/* Update scene-graph, including borders */
 | |
|  	wlr_scene_node_set_position(&c->scene->node, c->geom.x, c->geom.y);
 | |
| -	wlr_scene_node_set_position(&c->scene_surface->node, c->bw, c->bw);
 | |
| -	wlr_scene_rect_set_size(c->border[0], c->geom.width, c->bw);
 | |
| -	wlr_scene_rect_set_size(c->border[1], c->geom.width, c->bw);
 | |
| -	wlr_scene_rect_set_size(c->border[2], c->bw, c->geom.height - 2 * c->bw);
 | |
| -	wlr_scene_rect_set_size(c->border[3], c->bw, c->geom.height - 2 * c->bw);
 | |
| -	wlr_scene_node_set_position(&c->border[1]->node, 0, c->geom.height - c->bw);
 | |
| -	wlr_scene_node_set_position(&c->border[2]->node, 0, c->bw);
 | |
| -	wlr_scene_node_set_position(&c->border[3]->node, c->geom.width - c->bw, c->bw);
 | |
| +	wlr_scene_node_set_position(&c->scene_surface->node, c->bw, title_top ? th : c->bw);
 | |
| +	wlr_scene_rect_set_size(c->border[0], c->geom.width, (title_top && c->istabbed) ? 0 : c->bw);
 | |
| +	wlr_scene_rect_set_size(c->border[1], c->geom.width, (!title_top && c->istabbed) ? 0 : c->bw);
 | |
| +	wlr_scene_rect_set_size(c->border[2], c->bw, c->geom.height - (c->bw + th));
 | |
| +	wlr_scene_rect_set_size(c->border[3], c->bw, c->geom.height - (c->bw + th));
 | |
| +	wlr_scene_node_set_position(&c->border[1]->node, 0, c->geom.height - (title_top ? c->bw : th));
 | |
| +	wlr_scene_node_set_position(&c->border[2]->node, 0, title_top ? th : c->bw);
 | |
| +	wlr_scene_node_set_position(&c->border[3]->node, c->geom.width - c->bw, title_top ? th : c->bw);
 | |
|  
 | |
|  	/* this is a no-op if size hasn't changed */
 | |
|  	c->resize = client_set_size(c, c->geom.width - 2 * c->bw,
 | |
| -			c->geom.height - 2 * c->bw);
 | |
| +			c->geom.height - (c->bw + th));
 | |
|  	client_get_clip(c, &clip);
 | |
|  	wlr_scene_subsurface_tree_set_clip(&c->scene_surface->node, &clip);
 | |
|  }
 | |
| @@ -2274,6 +2506,11 @@ setup(void)
 | |
|  
 | |
|  	int i, sig[] = {SIGCHLD, SIGINT, SIGTERM, SIGPIPE};
 | |
|  	struct sigaction sa = {.sa_flags = SA_RESTART, .sa_handler = handlesig};
 | |
| +	cairo_surface_t *surface;
 | |
| +	cairo_t *cr;
 | |
| +	PangoFontDescription *desc;
 | |
| +	PangoLayout *layout;
 | |
| +	
 | |
|  	sigemptyset(&sa.sa_mask);
 | |
|  
 | |
|  	for (i = 0; i < (int)LENGTH(sig); i++)
 | |
| @@ -2506,6 +2743,24 @@ setup(void)
 | |
|  
 | |
|  	wlr_scene_set_presentation(scene, wlr_presentation_create(dpy, backend));
 | |
|  
 | |
| +	surface = cairo_image_surface_create(CAIRO_FORMAT_ARGB32, 0, 0);
 | |
| +
 | |
| +	cr = cairo_create(surface);
 | |
| +
 | |
| +	desc = pango_font_description_from_string(title_font);
 | |
| +	/* Create Pango layout. */
 | |
| +	layout = pango_cairo_create_layout(cr);
 | |
| +	pango_layout_set_font_description(layout, desc);
 | |
| +	pango_layout_set_text(layout, " ", -1);
 | |
| +	/* Set width and height to text size */
 | |
| +	pango_layout_get_pixel_size(layout, NULL, &title_height);
 | |
| +
 | |
| +	/* Cleanup */
 | |
| +	pango_font_description_free (desc);
 | |
| +	cairo_surface_destroy(surface);
 | |
| +	g_object_unref (layout);
 | |
| +	cairo_destroy(cr);
 | |
| +
 | |
|  	/* 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
 | |
|  	 * compositor has Xwayland support */
 | |
| @@ -2978,6 +3233,7 @@ createnotifyx11(struct wl_listener *listener, void *data)
 | |
|  	c->surface.xwayland = xsurface;
 | |
|  	c->type = X11;
 | |
|  	c->bw = borderpx;
 | |
| +	c->titleisinit = c->istabbed = 0;
 | |
|  
 | |
|  	/* Listen to the various events it can emit */
 | |
|  	LISTEN(&xsurface->events.associate, &c->associate, associatex11);
 | |
| -- 
 | |
| 2.44.0
 | |
| 
 | 
