Rebased branch to commit cc34dab of bar master.
This commit is contained in:
parent
55b657996f
commit
dcb0416af7
1
Makefile
1
Makefile
@ -33,6 +33,7 @@ debug: CC += ${CFDEBUG}
|
|||||||
clean:
|
clean:
|
||||||
rm -rf ./*.o
|
rm -rf ./*.o
|
||||||
rm -rf ./${EXEC}
|
rm -rf ./${EXEC}
|
||||||
|
rm -rf ./config.h
|
||||||
|
|
||||||
install: bar
|
install: bar
|
||||||
test -d ${DESTDIR}${BINDIR} || mkdir -p ${DESTDIR}${BINDIR}
|
test -d ${DESTDIR}${BINDIR} || mkdir -p ${DESTDIR}${BINDIR}
|
||||||
|
332
bar.c
332
bar.c
@ -40,12 +40,14 @@ enum {
|
|||||||
|
|
||||||
typedef struct screen_t {
|
typedef struct screen_t {
|
||||||
int x;
|
int x;
|
||||||
|
int y;
|
||||||
int width;
|
int width;
|
||||||
|
int height;
|
||||||
|
xcb_window_t window;
|
||||||
} screen_t;
|
} screen_t;
|
||||||
|
|
||||||
static xcb_connection_t *c;
|
static xcb_connection_t *c;
|
||||||
static xcb_screen_t *scr;
|
static xcb_screen_t *def_scr;
|
||||||
static xcb_window_t win;
|
|
||||||
static xcb_drawable_t canvas;
|
static xcb_drawable_t canvas;
|
||||||
static xcb_gcontext_t draw_gc;
|
static xcb_gcontext_t draw_gc;
|
||||||
static xcb_gcontext_t clear_gc;
|
static xcb_gcontext_t clear_gc;
|
||||||
@ -134,7 +136,7 @@ draw_char (screen_t *screen, int x, int align, wchar_t ch)
|
|||||||
(xcb_char2b_t *)&ch);
|
(xcb_char2b_t *)&ch);
|
||||||
|
|
||||||
/* Draw the underline */
|
/* Draw the underline */
|
||||||
if (BAR_UNDERLINE_HEIGHT)
|
if (BAR_UNDERLINE_HEIGHT)
|
||||||
xcb_fill_rect (underl_gc, x + screen->x, BAR_UNDERLINE*(BAR_HEIGHT-BAR_UNDERLINE_HEIGHT), ch_width, BAR_UNDERLINE_HEIGHT);
|
xcb_fill_rect (underl_gc, x + screen->x, BAR_UNDERLINE*(BAR_HEIGHT-BAR_UNDERLINE_HEIGHT), ch_width, BAR_UNDERLINE_HEIGHT);
|
||||||
|
|
||||||
return ch_width;
|
return ch_width;
|
||||||
@ -145,11 +147,13 @@ parse (char *text)
|
|||||||
{
|
{
|
||||||
char *p = text;
|
char *p = text;
|
||||||
|
|
||||||
|
int i;
|
||||||
int pos_x = 0;
|
int pos_x = 0;
|
||||||
int align = 0;
|
int align = 0;
|
||||||
screen_t *screen = &screens[0];
|
screen_t *screen = &screens[0];
|
||||||
|
|
||||||
xcb_fill_rect (clear_gc, 0, 0, bar_width, BAR_HEIGHT);
|
xcb_fill_rect (clear_gc, 0, 0, bar_width, BAR_HEIGHT);
|
||||||
|
|
||||||
for (;;) {
|
for (;;) {
|
||||||
if (*p == '\0')
|
if (*p == '\0')
|
||||||
return;
|
return;
|
||||||
@ -158,7 +162,7 @@ parse (char *text)
|
|||||||
|
|
||||||
if (*p == '\\' && p++ && *p != '\\' && strchr (control_characters, *p)) {
|
if (*p == '\\' && p++ && *p != '\\' && strchr (control_characters, *p)) {
|
||||||
switch (*p++) {
|
switch (*p++) {
|
||||||
case 'f':
|
case 'f':
|
||||||
xcb_set_fg (isdigit(*p) ? (*p)-'0' : 11);
|
xcb_set_fg (isdigit(*p) ? (*p)-'0' : 11);
|
||||||
p++;
|
p++;
|
||||||
break;
|
break;
|
||||||
@ -174,9 +178,17 @@ parse (char *text)
|
|||||||
if (num_screens == 1)
|
if (num_screens == 1)
|
||||||
break;
|
break;
|
||||||
if ((*p) == 'r') {
|
if ((*p) == 'r') {
|
||||||
screen = &screens[num_screens - 1];
|
for (i = num_screens-1; i >= 0; i--)
|
||||||
|
if (screens[i].width) {
|
||||||
|
screen = &screens[i];
|
||||||
|
break;
|
||||||
|
}
|
||||||
} else if ((*p) == 'l') {
|
} else if ((*p) == 'l') {
|
||||||
screen = &screens[0];
|
for (i = 0; i < num_screens; i++)
|
||||||
|
if (screens[i].width) {
|
||||||
|
screen = &screens[i];
|
||||||
|
break;
|
||||||
|
}
|
||||||
} else if ((*p) == 'n') {
|
} else if ((*p) == 'n') {
|
||||||
if (screen == &screens[num_screens - 1])
|
if (screen == &screens[num_screens - 1])
|
||||||
break;
|
break;
|
||||||
@ -187,7 +199,7 @@ parse (char *text)
|
|||||||
screen--;
|
screen--;
|
||||||
} else if (isdigit(*p)) {
|
} else if (isdigit(*p)) {
|
||||||
int index = (*p)-'0';
|
int index = (*p)-'0';
|
||||||
if (index < num_screens) {
|
if (index < num_screens && screens[index].width != 0) {
|
||||||
screen = &screens[index];
|
screen = &screens[index];
|
||||||
} else {
|
} else {
|
||||||
break;
|
break;
|
||||||
@ -211,8 +223,6 @@ parse (char *text)
|
|||||||
align = ALIGN_R;
|
align = ALIGN_R;
|
||||||
pos_x = 0;
|
pos_x = 0;
|
||||||
break;
|
break;
|
||||||
default:
|
|
||||||
break;
|
|
||||||
}
|
}
|
||||||
} else { /* utf-8 -> ucs-2 */
|
} else { /* utf-8 -> ucs-2 */
|
||||||
wchar_t t;
|
wchar_t t;
|
||||||
@ -315,7 +325,6 @@ set_ewmh_atoms ()
|
|||||||
xcb_intern_atom_cookie_t atom_cookie[atoms];
|
xcb_intern_atom_cookie_t atom_cookie[atoms];
|
||||||
xcb_atom_t atom_list[atoms];
|
xcb_atom_t atom_list[atoms];
|
||||||
xcb_intern_atom_reply_t *atom_reply;
|
xcb_intern_atom_reply_t *atom_reply;
|
||||||
int strut[12] = {0};
|
|
||||||
|
|
||||||
/* As suggested fetch all the cookies first (yum!) and then retrieve the
|
/* As suggested fetch all the cookies first (yum!) and then retrieve the
|
||||||
* atoms to exploit the async'ness */
|
* atoms to exploit the async'ness */
|
||||||
@ -331,27 +340,44 @@ set_ewmh_atoms ()
|
|||||||
}
|
}
|
||||||
|
|
||||||
/* Prepare the strut array */
|
/* Prepare the strut array */
|
||||||
if (bar_bottom) {
|
for (screen_t *cur_screen = screens; cur_screen < screens+num_screens; cur_screen++) {
|
||||||
strut[3] = BAR_HEIGHT;
|
int strut[12] = {0};
|
||||||
strut[11] = bar_width;
|
if (cur_screen->width == 0)
|
||||||
} else {
|
continue;
|
||||||
strut[2] = BAR_HEIGHT;
|
if (bar_bottom) {
|
||||||
strut[9] = bar_width;
|
strut[3] = BAR_HEIGHT;
|
||||||
}
|
strut[10] = cur_screen->x;
|
||||||
|
strut[11] = cur_screen->x + cur_screen->width;
|
||||||
|
} else {
|
||||||
|
strut[2] = BAR_HEIGHT;
|
||||||
|
strut[8] = cur_screen->x;
|
||||||
|
strut[9] = cur_screen->x + cur_screen->width;
|
||||||
|
}
|
||||||
|
|
||||||
xcb_change_property (c, XCB_PROP_MODE_REPLACE, win, atom_list[NET_WM_WINDOW_OPACITY], XCB_ATOM_CARDINAL, 32, 1, (const uint32_t []){ (uint32_t)(BAR_OPACITY * 0xffffffff) } );
|
xcb_change_property (c, XCB_PROP_MODE_REPLACE, cur_screen->window, atom_list[NET_WM_WINDOW_OPACITY], XCB_ATOM_CARDINAL, 32, 1, (const uint32_t []){ (uint32_t)(BAR_OPACITY * 0xffffffff) } );
|
||||||
xcb_change_property (c, XCB_PROP_MODE_REPLACE, win, atom_list[NET_WM_WINDOW_TYPE], XCB_ATOM_ATOM, 32, 1, &atom_list[NET_WM_WINDOW_TYPE_DOCK]);
|
xcb_change_property (c, XCB_PROP_MODE_REPLACE, cur_screen->window, atom_list[NET_WM_WINDOW_TYPE], XCB_ATOM_ATOM, 32, 1, &atom_list[NET_WM_WINDOW_TYPE_DOCK]);
|
||||||
xcb_change_property (c, XCB_PROP_MODE_APPEND, win, atom_list[NET_WM_STATE], XCB_ATOM_ATOM, 32, 2, &atom_list[NET_WM_STATE_STICKY]);
|
xcb_change_property (c, XCB_PROP_MODE_APPEND, cur_screen->window, atom_list[NET_WM_STATE], XCB_ATOM_ATOM, 32, 2, &atom_list[NET_WM_STATE_STICKY]);
|
||||||
xcb_change_property (c, XCB_PROP_MODE_REPLACE, win, atom_list[NET_WM_DESKTOP], XCB_ATOM_CARDINAL, 32, 1, (const uint32_t []){ -1 } );
|
xcb_change_property (c, XCB_PROP_MODE_REPLACE, cur_screen->window, atom_list[NET_WM_DESKTOP], XCB_ATOM_CARDINAL, 32, 1, (const uint32_t []){ -1 } );
|
||||||
xcb_change_property (c, XCB_PROP_MODE_REPLACE, win, atom_list[NET_WM_STRUT_PARTIAL], XCB_ATOM_CARDINAL, 32, 12, strut);
|
xcb_change_property (c, XCB_PROP_MODE_REPLACE, cur_screen->window, atom_list[NET_WM_STRUT_PARTIAL], XCB_ATOM_CARDINAL, 32, 12, strut);
|
||||||
xcb_change_property (c, XCB_PROP_MODE_REPLACE, win, atom_list[NET_WM_STRUT], XCB_ATOM_CARDINAL, 32, 4, strut);
|
xcb_change_property (c, XCB_PROP_MODE_REPLACE, cur_screen->window, atom_list[NET_WM_STRUT], XCB_ATOM_CARDINAL, 32, 4, strut);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
xcb_window_t
|
||||||
|
create_window(xcb_window_t root, int x, int y, int width, int height, xcb_visualid_t visual) {
|
||||||
|
xcb_window_t window = xcb_generate_id(c);
|
||||||
|
xcb_create_window(c, XCB_COPY_FROM_PARENT, window, root, x, y, width, height, 0, XCB_WINDOW_CLASS_INPUT_OUTPUT, visual, XCB_CW_BACK_PIXEL | XCB_CW_EVENT_MASK, (const uint32_t []){ palette[10], XCB_EVENT_MASK_EXPOSURE });
|
||||||
|
|
||||||
|
xcb_change_window_attributes (c, window, XCB_CW_OVERRIDE_REDIRECT, (const uint32_t []){ force_docking });
|
||||||
|
|
||||||
|
return window;
|
||||||
}
|
}
|
||||||
|
|
||||||
int
|
int
|
||||||
get_randr_outputs(xcb_window_t w, screen_t **spp)
|
get_randr_outputs(xcb_window_t w, xcb_rectangle_t **rects)
|
||||||
{
|
{
|
||||||
int i, j, num, cnt = 0;
|
int i, j, num, cnt = 0;
|
||||||
screen_t *s;
|
xcb_rectangle_t *r;
|
||||||
xcb_generic_error_t *err;
|
xcb_generic_error_t *err;
|
||||||
xcb_randr_get_screen_resources_current_cookie_t rres_query;
|
xcb_randr_get_screen_resources_current_cookie_t rres_query;
|
||||||
xcb_randr_get_screen_resources_current_reply_t *rres_reply;
|
xcb_randr_get_screen_resources_current_reply_t *rres_reply;
|
||||||
@ -362,35 +388,37 @@ get_randr_outputs(xcb_window_t w, screen_t **spp)
|
|||||||
rres_reply = xcb_randr_get_screen_resources_current_reply(c, rres_query, &err);
|
rres_reply = xcb_randr_get_screen_resources_current_reply(c, rres_query, &err);
|
||||||
if (rres_reply == NULL || err != NULL) {
|
if (rres_reply == NULL || err != NULL) {
|
||||||
fprintf(stderr, "Failed to get current randr screen resources\n");
|
fprintf(stderr, "Failed to get current randr screen resources\n");
|
||||||
|
free(rres_reply);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
num = xcb_randr_get_screen_resources_current_outputs_length(rres_reply);
|
num = xcb_randr_get_screen_resources_current_outputs_length(rres_reply);
|
||||||
outputs = xcb_randr_get_screen_resources_current_outputs(rres_reply);
|
outputs = xcb_randr_get_screen_resources_current_outputs(rres_reply);
|
||||||
config_timestamp = rres_reply->config_timestamp;
|
config_timestamp = rres_reply->config_timestamp;
|
||||||
free(rres_reply);
|
free(rres_reply);
|
||||||
if (num < 1)
|
if (num < 1) {
|
||||||
|
fprintf(stderr, "Failed to get current randr outputs\n");
|
||||||
return 0;
|
return 0;
|
||||||
|
}
|
||||||
xcb_rectangle_t temp[num];
|
xcb_rectangle_t temp[num];
|
||||||
|
|
||||||
/* use all outputs */
|
/* get all outputs */
|
||||||
for (i = 0; i < num; i++) {
|
for (i = 0; i < num; i++) {
|
||||||
xcb_randr_get_output_info_cookie_t output_cookie;
|
xcb_randr_get_output_info_cookie_t output_query;
|
||||||
xcb_randr_get_output_info_reply_t *output_reply;
|
xcb_randr_get_output_info_reply_t *output_reply;
|
||||||
xcb_randr_get_crtc_info_cookie_t crtc_cookie;
|
xcb_randr_get_crtc_info_cookie_t crtc_query;
|
||||||
xcb_randr_get_crtc_info_reply_t *crtc_reply;
|
xcb_randr_get_crtc_info_reply_t *crtc_reply;
|
||||||
|
|
||||||
output_cookie = xcb_randr_get_output_info(c, outputs[i], config_timestamp);
|
output_query = xcb_randr_get_output_info(c, outputs[i], config_timestamp);
|
||||||
output_reply = xcb_randr_get_output_info_reply(c, output_cookie, &err);
|
output_reply = xcb_randr_get_output_info_reply(c, output_query, &err);
|
||||||
if (err != NULL || output_reply == NULL || output_reply->crtc == XCB_NONE) {
|
if (err != NULL || output_reply == NULL || output_reply->crtc == XCB_NONE) {
|
||||||
temp[i].width = 0;
|
temp[i].width = 0;
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
crtc_cookie = xcb_randr_get_crtc_info(c, output_reply->crtc, config_timestamp);
|
crtc_query = xcb_randr_get_crtc_info(c, output_reply->crtc, config_timestamp);
|
||||||
crtc_reply = xcb_randr_get_crtc_info_reply(c, crtc_cookie, &err);
|
crtc_reply = xcb_randr_get_crtc_info_reply(c, crtc_query, &err);
|
||||||
if (err != NULL || crtc_reply == NULL) {
|
if (err != NULL | crtc_reply == NULL) {
|
||||||
fprintf(stderr, "Failed to get randr crtc info\n");
|
fprintf(stderr, "Failed to get randr ctrc info\n");
|
||||||
temp[i].width = 0;
|
temp[i].width = 0;
|
||||||
free(output_reply);
|
free(output_reply);
|
||||||
continue;
|
continue;
|
||||||
@ -403,13 +431,13 @@ get_randr_outputs(xcb_window_t w, screen_t **spp)
|
|||||||
free(output_reply);
|
free(output_reply);
|
||||||
cnt++;
|
cnt++;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (cnt < 1) {
|
if (cnt < 1) {
|
||||||
fprintf(stderr, "No usable randr outputs\n");
|
fprintf(stderr, "No usable randr outputs\n");
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* check for clones */
|
/* check for clones and inactive outputs */
|
||||||
for (i = 0; i < num; i++) {
|
for (i = 0; i < num; i++) {
|
||||||
if (temp[i].width == 0)
|
if (temp[i].width == 0)
|
||||||
continue;
|
continue;
|
||||||
@ -429,28 +457,85 @@ get_randr_outputs(xcb_window_t w, screen_t **spp)
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
s = calloc(cnt, sizeof(screen_t));
|
r = calloc(cnt, sizeof(xcb_rectangle_t));
|
||||||
|
if (r == NULL)
|
||||||
|
exit(1);
|
||||||
|
|
||||||
|
for (i = j = 0; i < num && j < cnt; i++) {
|
||||||
|
if (temp[i].width) {
|
||||||
|
r[j].x = temp[i].x;
|
||||||
|
r[j].y = temp[i].y;
|
||||||
|
r[j].width = temp[i].width;
|
||||||
|
r[j++].height = temp[i].height;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
*rects = r;
|
||||||
|
|
||||||
|
return cnt;
|
||||||
|
}
|
||||||
|
|
||||||
|
screen_t
|
||||||
|
*screens_adjust(xcb_rectangle_t *rects, int nrects)
|
||||||
|
{
|
||||||
|
int i, y;
|
||||||
|
screen_t *s;
|
||||||
|
|
||||||
|
s = calloc(nrects, sizeof(screen_t));
|
||||||
if (s == NULL)
|
if (s == NULL)
|
||||||
exit(1);
|
exit(1);
|
||||||
|
|
||||||
for (i = j = 0; i < num && j < cnt; i++)
|
for (i = 0; i < nrects; i++) {
|
||||||
if (temp[i].width) {
|
s[i].x = rects[i].x;
|
||||||
s[j].x = temp[i].x;
|
s[i].y = rects[i].y;
|
||||||
s[j++].width = temp[i].width;
|
s[i].width = rects[i].width;
|
||||||
|
s[i].height = rects[i].height;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Add BAR_OFFSET to the last screen */
|
||||||
|
int right_bar_offset = def_scr->width_in_pixels - bar_width - BAR_OFFSET;
|
||||||
|
for (i = nrects-1; i >= 0 && right_bar_offset > 0; i--) {
|
||||||
|
if (right_bar_offset >= s[i].width) {
|
||||||
|
right_bar_offset -= s[i].width;
|
||||||
|
/* Deactivate the screen */
|
||||||
|
s[i].width = 0;
|
||||||
|
} else {
|
||||||
|
s[i].width -= right_bar_offset;
|
||||||
|
right_bar_offset = 0;
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
*spp = s;
|
/* Subtract BAR_OFFSET from the first screen */
|
||||||
|
int left_bar_offset = BAR_OFFSET;
|
||||||
|
for (i = 0; i < nrects && left_bar_offset > 0; i++) {
|
||||||
|
if (left_bar_offset >= s[i].width) {
|
||||||
|
left_bar_offset -= s[i].width;
|
||||||
|
s[i].width = 0;
|
||||||
|
} else {
|
||||||
|
s[i].x = left_bar_offset;
|
||||||
|
s[i].width -= left_bar_offset;
|
||||||
|
left_bar_offset = 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
return cnt;
|
/* Create windows for each output */
|
||||||
|
for (i = 0; i < num_screens; i++) {
|
||||||
|
if (s[i].width) {
|
||||||
|
y = (bar_bottom ? ( s[i].height - BAR_HEIGHT ) : 0 ) + s[i].y;
|
||||||
|
s[i].window = create_window(def_scr->root, s[i].x, y, s[i].width,
|
||||||
|
BAR_HEIGHT, def_scr->root_visual);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return s;
|
||||||
}
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
init (void)
|
init (void)
|
||||||
{
|
{
|
||||||
|
int i, nrects;
|
||||||
xcb_window_t root;
|
xcb_window_t root;
|
||||||
int y;
|
xcb_rectangle_t *rects = NULL;
|
||||||
// const xcb_query_extension_reply_t *randr_ext_reply;
|
|
||||||
// const xcb_query_extension_reply_t *xinerama_ext_reply;
|
|
||||||
const xcb_query_extension_reply_t *ext_reply;
|
const xcb_query_extension_reply_t *ext_reply;
|
||||||
|
|
||||||
/* Connect to X */
|
/* Connect to X */
|
||||||
@ -461,49 +546,48 @@ init (void)
|
|||||||
}
|
}
|
||||||
|
|
||||||
/* Grab infos from the first screen */
|
/* Grab infos from the first screen */
|
||||||
scr = xcb_setup_roots_iterator (xcb_get_setup (c)).data;
|
def_scr = xcb_setup_roots_iterator (xcb_get_setup (c)).data;
|
||||||
root = scr->root;
|
root = def_scr->root;
|
||||||
|
|
||||||
/* where to place the window */
|
/* where to place the window */
|
||||||
y = (bar_bottom) ? (scr->height_in_pixels - BAR_HEIGHT) : 0;
|
bar_width = (BAR_WIDTH < 0) ? (def_scr->width_in_pixels - BAR_OFFSET) : BAR_WIDTH;
|
||||||
bar_width = (BAR_WIDTH < 0) ? (scr->width_in_pixels - BAR_OFFSET) : BAR_WIDTH;
|
|
||||||
|
|
||||||
/* Load the font */
|
/* Load the font */
|
||||||
if (font_load ((const char* []){ BAR_FONT }))
|
if (font_load ((const char* []){ BAR_FONT }))
|
||||||
exit (1);
|
exit (1);
|
||||||
|
|
||||||
/* Generate a list of screens */
|
/* Generate a list of screens */
|
||||||
num_screens = 0;
|
ext_reply = xcb_get_extension_data(c, &xcb_randr_id);
|
||||||
if ((ext_reply = xcb_get_extension_data(c, &xcb_randr_id)) && ext_reply->present) {
|
if (ext_reply != NULL && ext_reply->present) {
|
||||||
num_screens = get_randr_outputs(root, &screens);
|
nrects = get_randr_outputs(root, &rects);
|
||||||
if (num_screens)
|
if (nrects)
|
||||||
randr_event = ext_reply->first_event;
|
randr_event = ext_reply->first_event;
|
||||||
}
|
} else {
|
||||||
else if ((ext_reply = xcb_get_extension_data(c, &xcb_xinerama_id)) && ext_reply->present) {
|
ext_reply = xcb_get_extension_data(c, &xcb_xinerama_id);
|
||||||
xcb_xinerama_is_active_cookie_t xia_query;
|
if (ext_reply != NULL && ext_reply->present) {
|
||||||
xcb_xinerama_is_active_reply_t *xia_reply;
|
xcb_xinerama_is_active_cookie_t xia_query;
|
||||||
xcb_xinerama_query_screens_cookie_t xqs_query;
|
xcb_xinerama_is_active_reply_t *xia_reply;
|
||||||
xcb_xinerama_query_screens_reply_t *xqs_reply;
|
|
||||||
xcb_xinerama_screen_info_t *xs_info;
|
|
||||||
|
|
||||||
xia_query = xcb_xinerama_is_active(c);
|
xia_query = xcb_xinerama_is_active(c);
|
||||||
xia_reply = xcb_xinerama_is_active_reply(c, xia_query, NULL);
|
xia_reply = xcb_xinerama_is_active_reply(c, xia_query, NULL);
|
||||||
|
if (xia_reply != NULL && xia_reply->state) {
|
||||||
|
xcb_xinerama_query_screens_cookie_t xqs_query;
|
||||||
|
xcb_xinerama_query_screens_reply_t *xqs_reply;
|
||||||
|
xcb_xinerama_screen_info_t *xs_info;
|
||||||
|
|
||||||
if (xia_reply) {
|
|
||||||
if (xia_reply->state) {
|
|
||||||
xqs_query = xcb_xinerama_query_screens(c);
|
xqs_query = xcb_xinerama_query_screens(c);
|
||||||
xqs_reply = xcb_xinerama_query_screens_reply(c, xqs_query, NULL);
|
xqs_reply = xcb_xinerama_query_screens_reply(c, xqs_query, NULL);
|
||||||
if (xqs_reply) {
|
if (xqs_reply) {
|
||||||
|
nrects = xcb_xinerama_query_screens_screen_info_length(xqs_reply);
|
||||||
num_screens = xcb_xinerama_query_screens_screen_info_length(xqs_reply);
|
|
||||||
xs_info = xcb_xinerama_query_screens_screen_info(xqs_reply);
|
xs_info = xcb_xinerama_query_screens_screen_info(xqs_reply);
|
||||||
screens = calloc (num_screens, sizeof(screen_t));
|
rects = calloc(nrects, sizeof(xcb_rectangle_t));
|
||||||
if (screens == NULL)
|
if (rects == NULL)
|
||||||
exit (1);
|
exit(1);
|
||||||
|
for (i = 0; i < nrects; i++) {
|
||||||
for (int i = 0; i < num_screens; i++) {
|
rects[i].x = xs_info[i].x_org;
|
||||||
screens[i].x = xs_info[i].x_org;
|
rects[i].y = xs_info[i].y_org;
|
||||||
screens[i].width = xs_info[i].width;
|
rects[i].width = xs_info[i].width;
|
||||||
|
rects[i].height = xs_info[i].height;
|
||||||
}
|
}
|
||||||
free(xqs_reply);
|
free(xqs_reply);
|
||||||
}
|
}
|
||||||
@ -512,29 +596,31 @@ init (void)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (num_screens == 0) {
|
if (nrects) {
|
||||||
|
num_screens = nrects;
|
||||||
|
screens = screens_adjust(rects, nrects);
|
||||||
|
free(rects);
|
||||||
|
} else {
|
||||||
|
/* no randr or xinerama outputs - default to single screen */
|
||||||
num_screens = 1;
|
num_screens = 1;
|
||||||
screens = calloc(1, sizeof(screen_t));
|
screens = calloc(1, sizeof(screen_t));
|
||||||
if (screens == NULL)
|
if (screens == NULL)
|
||||||
exit(1);
|
exit(1);
|
||||||
screens[0].x = 0;
|
screens->x = BAR_OFFSET;
|
||||||
screens[0].width = bar_width;
|
screens->y = 0;
|
||||||
|
screens->width = bar_width - BAR_OFFSET;
|
||||||
|
screens->height = def_scr->height_in_pixels;
|
||||||
|
int y = bar_bottom ? (screens->height - BAR_HEIGHT) : 0;
|
||||||
|
screens->window = create_window(root, screens->x, y,
|
||||||
|
screens->width, BAR_HEIGHT, def_scr->root_visual);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Create the main window */
|
|
||||||
win = xcb_generate_id (c);
|
|
||||||
xcb_create_window (c, XCB_COPY_FROM_PARENT, win, root, BAR_OFFSET, y, bar_width,
|
|
||||||
BAR_HEIGHT, 0, XCB_WINDOW_CLASS_INPUT_OUTPUT, scr->root_visual,
|
|
||||||
XCB_CW_BACK_PIXEL | XCB_CW_EVENT_MASK, (const uint32_t []){ palette[10], XCB_EVENT_MASK_EXPOSURE });
|
|
||||||
|
|
||||||
/* For WM that support EWMH atoms */
|
/* For WM that support EWMH atoms */
|
||||||
set_ewmh_atoms();
|
set_ewmh_atoms();
|
||||||
|
|
||||||
xcb_change_window_attributes (c, win, XCB_CW_OVERRIDE_REDIRECT, (const uint32_t []){ force_docking });
|
|
||||||
|
|
||||||
/* Create a temporary canvas */
|
/* Create a temporary canvas */
|
||||||
canvas = xcb_generate_id (c);
|
canvas = xcb_generate_id (c);
|
||||||
xcb_create_pixmap (c, scr->root_depth, canvas, root, bar_width, BAR_HEIGHT);
|
xcb_create_pixmap (c, def_scr->root_depth, canvas, root, bar_width, BAR_HEIGHT);
|
||||||
|
|
||||||
/* Create the gc for drawing */
|
/* Create the gc for drawing */
|
||||||
draw_gc = xcb_generate_id (c);
|
draw_gc = xcb_generate_id (c);
|
||||||
@ -547,7 +633,9 @@ init (void)
|
|||||||
xcb_create_gc (c, underl_gc, root, XCB_GC_FOREGROUND, (const uint32_t []){ palette[10] });
|
xcb_create_gc (c, underl_gc, root, XCB_GC_FOREGROUND, (const uint32_t []){ palette[10] });
|
||||||
|
|
||||||
/* Make the bar visible */
|
/* Make the bar visible */
|
||||||
xcb_map_window (c, win);
|
for (i = 0; i < num_screens; i++)
|
||||||
|
if (screens[i].width)
|
||||||
|
xcb_map_window(c, screens[i].window);
|
||||||
|
|
||||||
xcb_flush (c);
|
xcb_flush (c);
|
||||||
}
|
}
|
||||||
@ -560,12 +648,14 @@ cleanup (void)
|
|||||||
if (fontset[i].xcb_ft)
|
if (fontset[i].xcb_ft)
|
||||||
xcb_close_font (c, fontset[i].xcb_ft);
|
xcb_close_font (c, fontset[i].xcb_ft);
|
||||||
}
|
}
|
||||||
if (screens)
|
if (screens) {
|
||||||
|
for(screen_t *cur_screen = screens; cur_screen < screens + num_screens; cur_screen++) {
|
||||||
|
xcb_destroy_window( c, cur_screen->window );
|
||||||
|
}
|
||||||
free (screens);
|
free (screens);
|
||||||
|
}
|
||||||
if (canvas)
|
if (canvas)
|
||||||
xcb_free_pixmap (c, canvas);
|
xcb_free_pixmap (c, canvas);
|
||||||
if (win)
|
|
||||||
xcb_destroy_window (c, win);
|
|
||||||
if (draw_gc)
|
if (draw_gc)
|
||||||
xcb_free_gc (c, draw_gc);
|
xcb_free_gc (c, draw_gc);
|
||||||
if (clear_gc)
|
if (clear_gc)
|
||||||
@ -586,18 +676,41 @@ sighandle (int signal)
|
|||||||
void
|
void
|
||||||
handle_randr_event (xcb_generic_event_t *ev)
|
handle_randr_event (xcb_generic_event_t *ev)
|
||||||
{
|
{
|
||||||
int num;
|
int i, j, nrects;
|
||||||
screen_t *s, *t = screens;
|
xcb_rectangle_t *rects;
|
||||||
|
screen_t *s, *t;
|
||||||
|
|
||||||
num = get_randr_outputs(scr->root, &s);
|
nrects = get_randr_outputs(def_scr->root, &rects);
|
||||||
if (num <= 0) {
|
|
||||||
fprintf(stderr, "handle_randr_event: no outputs\n");
|
t = screens;
|
||||||
exit(1);
|
s = screens_adjust(rects, nrects);
|
||||||
|
free(rects);
|
||||||
|
|
||||||
|
for (i = 0; i < nrects; i++) {
|
||||||
|
if (s[i].width == 0)
|
||||||
|
continue;
|
||||||
|
for (j = 0; j < num_screens; j++)
|
||||||
|
if (t[j].width && t[j].window &&
|
||||||
|
s[i].x == t[j].x && s[i].y == t[j].y &&
|
||||||
|
s[i].width == t[j].width && s[i].height == t[j].height) {
|
||||||
|
xcb_destroy_window(c, s[i].window);
|
||||||
|
s[i].window = t[j].window;
|
||||||
|
t[j].window = 0;
|
||||||
|
break;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
/* there needs to be more fixups after the screens are returned */
|
|
||||||
screens = s;
|
for (i = 0; i < num_screens; i++)
|
||||||
num_screens = num;
|
if (t[i].window)
|
||||||
|
xcb_destroy_window(c, t[i].window);
|
||||||
|
|
||||||
|
for (i = 0; i < nrects; i++)
|
||||||
|
if (s[i].width)
|
||||||
|
xcb_map_window(c, s[i].window);
|
||||||
|
|
||||||
free(t);
|
free(t);
|
||||||
|
num_screens = nrects;
|
||||||
|
screens = s;
|
||||||
}
|
}
|
||||||
|
|
||||||
int
|
int
|
||||||
@ -657,13 +770,11 @@ main (int argc, char **argv)
|
|||||||
while ((ev = xcb_poll_for_event (c))) {
|
while ((ev = xcb_poll_for_event (c))) {
|
||||||
int type = ev->response_type & 0x7f;
|
int type = ev->response_type & 0x7f;
|
||||||
|
|
||||||
if (randr_event > -1 && type == randr_event + XCB_RANDR_SCREEN_CHANGE_NOTIFY)
|
if (type == XCB_EXPOSE)
|
||||||
handle_randr_event(ev);
|
if (((xcb_expose_event_t *)ev)->count == 0)
|
||||||
else if (type == XCB_EXPOSE) {
|
|
||||||
expose_ev = (xcb_expose_event_t *)ev;
|
|
||||||
if (expose_ev->count == 0)
|
|
||||||
redraw = 1;
|
redraw = 1;
|
||||||
}
|
else if (randr_event > -1 && type == randr_event + XCB_RANDR_SCREEN_CHANGE_NOTIFY)
|
||||||
|
handle_randr_event(ev);
|
||||||
|
|
||||||
free (ev);
|
free (ev);
|
||||||
}
|
}
|
||||||
@ -671,7 +782,10 @@ main (int argc, char **argv)
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (redraw) /* Copy our temporary pixmap onto the window */
|
if (redraw) /* Copy our temporary pixmap onto the window */
|
||||||
xcb_copy_area (c, canvas, win, draw_gc, 0, 0, 0, 0, bar_width, BAR_HEIGHT);
|
for (screen_t* cur_screen = screens; cur_screen < screens + num_screens; cur_screen++)
|
||||||
|
if (cur_screen->width)
|
||||||
|
xcb_copy_area (c, canvas, cur_screen->window, draw_gc, cur_screen->x,
|
||||||
|
0, 0, 0, cur_screen->width, BAR_HEIGHT);
|
||||||
|
|
||||||
xcb_flush (c);
|
xcb_flush (c);
|
||||||
}
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user