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
120
121
122
123
124
125
126
127
128
|
/*
* x11_support.c
*
* Created by buzzert <buzzert@buzzert.net> 2019-01-18
*/
#include "x11_support.h"
#include <X11/extensions/Xrandr.h>
#include <stdio.h>
#include <stdlib.h>
static Window __window = { 0 };
static Display *__display = NULL;
static void x11_get_display_bounds_w(Window window, unsigned int monitor_num, x11_display_bounds_t *out_bounds);
static Window get_window_from_environment_or_make_one(Display *display, int *out_width, int *out_height)
{
Window window;
Window root_window = DefaultRootWindow(__display);
const char *env_window = getenv("XSCREENSAVER_WINDOW");
if (env_window != NULL && env_window[0] != 0) {
char *endptr = NULL;
unsigned long long number = strtoull(env_window, &endptr, 0);
root_window = (Window)number;
}
// Figure out which monitor this is supposed to go on
const unsigned int preferred_monitor = get_preferred_monitor_num();
x11_display_bounds_t bounds;
x11_get_display_bounds_w(root_window, preferred_monitor, &bounds);
window = XCreateSimpleWindow(
display, // display
root_window, // parent window
bounds.x, // x
bounds.y, // y
bounds.width, // width
bounds.height, // height
0, // border_width
0, // border
0 // background
);
*out_width = bounds.width;
*out_height = bounds.height;
return window;
}
void x11_get_display_bounds(unsigned int monitor_num, x11_display_bounds_t *out_bounds)
{
x11_get_display_bounds_w(__window, monitor_num, out_bounds);
}
static void x11_get_display_bounds_w(Window window, unsigned int monitor_num, x11_display_bounds_t *out_bounds)
{
int num_monitors = 0;
XRRMonitorInfo *monitor_infos = XRRGetMonitors(__display, window, True, &num_monitors);
if (num_monitors == 0) {
fprintf(stderr, "FATAL: Couldn't get monitor info from XRandR!\n");
exit(1);
}
unsigned int idx = monitor_num;
if (idx > num_monitors) {
fprintf(stderr, "WARNING: Specified monitor number is greater than the number of connected monitors!\n");
idx = 0;
}
XRRMonitorInfo *monitor = &monitor_infos[idx];
out_bounds->x = monitor->x;
out_bounds->y = monitor->y;
out_bounds->width = monitor->width;
out_bounds->height = monitor->height;
}
unsigned int get_preferred_monitor_num()
{
const char *preferred_monitor = getenv("BUZZLOCKER_MONITOR_NUM");
if (preferred_monitor != NULL && preferred_monitor[0] != 0) {
char *endptr = NULL;
unsigned long int result = strtoul(preferred_monitor, &endptr, 0);
return result;
}
return 0;
}
cairo_surface_t* x11_helper_acquire_cairo_surface()
{
__display = XOpenDisplay(NULL);
if (__display == NULL) {
fprintf(stderr, "Error opening display\n");
return NULL;
}
// Create (or get) window
int width, height;
__window = get_window_from_environment_or_make_one(__display, &width, &height);
// Enable key events
XSelectInput(__display, __window, ButtonPressMask | KeyPressMask | StructureNotifyMask);
// Map window to display
XMapWindow(__display, __window);
// Create cairo surface
int screen = DefaultScreen(__display);
Visual *visual = DefaultVisual(__display, screen);
cairo_surface_t *surface = cairo_xlib_surface_create(
__display,
__window,
visual,
width,
height
);
return surface;
}
void x11_helper_destroy_surface(cairo_surface_t *surface)
{
cairo_surface_destroy(surface);
XCloseDisplay(__display);
}
|