aboutsummaryrefslogtreecommitdiffstats
path: root/sway/debug-tree.c
blob: ae0a1869b0738cf0c14faff45d4aa3935eb00d66 (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
#include <pango/pangocairo.h>
#include <wlr/backend.h>
#include <wlr/render/wlr_texture.h>
#include <wlr/util/log.h>
#include "config.h"
#include "sway/input/input-manager.h"
#include "sway/input/seat.h"
#include "sway/server.h"
#include "sway/tree/container.h"
#include "sway/tree/layout.h"
#include "cairo.h"
#include "config.h"
#include "pango.h"

static const char *layout_to_str(enum sway_container_layout layout) {
	switch (layout) {
	case L_HORIZ:
		return "L_HORIZ";
	case L_VERT:
		return "L_VERT";
	case L_STACKED:
		return "L_STACKED";
	case L_TABBED:
		return "L_TABBED";
	case L_FLOATING:
		return "L_FLOATING";
	case L_NONE:
	default:
		return "L_NONE";
	}
}

static int draw_container(cairo_t *cairo, struct sway_container *container,
		struct sway_container *focus, int x, int y) {
	int text_width, text_height;
	get_text_size(cairo, "monospace", &text_width, &text_height,
		1, false, "%s id:%zd '%s' %s %.fx%.f@%.f,%.f",
		container_type_to_str(container->type), container->id, container->name,
		layout_to_str(container->layout),
		container->width, container->height, container->x, container->y);
	cairo_save(cairo);
	cairo_rectangle(cairo, x + 2, y, text_width - 2, text_height);
	cairo_set_source_u32(cairo, 0xFFFFFFE0);
	cairo_fill(cairo);
	int height = text_height;
	if (container->children) {
		for (int i = 0; i < container->children->length; ++i) {
			struct sway_container *child = container->children->items[i];
			if (child->parent == container) {
				cairo_set_source_u32(cairo, 0x000000FF);
			} else {
				cairo_set_source_u32(cairo, 0xFF0000FF);
			}
			height += draw_container(cairo, child, focus, x + 10, y + height);
		}
	}
	cairo_set_source_u32(cairo, 0xFFFFFFE0);
	cairo_rectangle(cairo, x, y, 2, height);
	cairo_fill(cairo);
	cairo_restore(cairo);
	cairo_move_to(cairo, x, y);
	if (focus == container) {
		cairo_set_source_u32(cairo, 0x0000FFFF);
	}
	pango_printf(cairo, "monospace", 1, false, "%s id:%zd '%s' %s %.fx%.f@%.f,%.f",
		container_type_to_str(container->type), container->id, container->name,
		layout_to_str(container->layout),
		container->width, container->height, container->x, container->y);
	return height;
}

bool enable_debug_tree = false;

void update_debug_tree() {
	if (!enable_debug_tree) {
		return;
	}

	int width = 640, height = 480;
	for (int i = 0; i < root_container.children->length; ++i) {
		struct sway_container *container = root_container.children->items[i];
		if (container->width > width) {
			width = container->width;
		}
		if (container->height > height) {
			height = container->height;
		}
	}
	cairo_surface_t *surface =
		cairo_image_surface_create(CAIRO_FORMAT_ARGB32, width, height);
	cairo_t *cairo = cairo_create(surface);
	PangoContext *pango = pango_cairo_create_context(cairo);

	struct sway_seat *seat = NULL;
	wl_list_for_each(seat, &input_manager->seats, link) {
		break;
	}

	struct sway_container *focus = NULL;
	if (seat != NULL) {
		focus = seat_get_focus(seat);
	}
	cairo_set_source_u32(cairo, 0x000000FF);
	draw_container(cairo, &root_container, focus, 0, 0);

	cairo_surface_flush(surface);
	struct wlr_renderer *renderer = wlr_backend_get_renderer(server.backend);
	if (root_container.sway_root->debug_tree) {
		wlr_texture_destroy(root_container.sway_root->debug_tree);
	}
	unsigned char *data = cairo_image_surface_get_data(surface);
	int stride = cairo_format_stride_for_width(CAIRO_FORMAT_ARGB32, width);
	struct wlr_texture *texture = wlr_texture_from_pixels(renderer,
		WL_SHM_FORMAT_ARGB8888, stride, width, height, data);
	root_container.sway_root->debug_tree = texture;
	cairo_surface_destroy(surface);
	g_object_unref(pango);
	cairo_destroy(cairo);
}