/* * common/xscreen.c - common X screen management * * Copyright © 2007-2008 Julien Danjou * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License along * with this program; if not, write to the Free Software Foundation, Inc., * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. * */ #include #include #include #include "common/xscreen.h" /** Return the Xinerama screen number where the coordinates belongs to * \param disp Display ref * \param x x coordinate of the window * \param y y coordinate of the window * \return screen number or DefaultScreen of disp on no match */ int screen_get_bycoord(ScreensInfo *si, int screen, int x, int y) { int i; /* don't waste our time */ if(!si->xinerama_is_active) return screen; for(i = 0; i < si->nscreen; i++) if((x < 0 || (x >= si->geometry[i].x && x < si->geometry[i].x + si->geometry[i].width)) && (y < 0 || (y >= si->geometry[i].y && y < si->geometry[i].y + si->geometry[i].height))) return i; return screen; } static inline area_t screen_xsitoarea(xcb_xinerama_screen_info_t si) { area_t a; a.x = si.x_org; a.y = si.y_org; a.width = si.width; a.height = si.height; a.next = a.prev = NULL; return a; } void screensinfo_delete(ScreensInfo **si) { p_delete(&(*si)->geometry); p_delete(si); } static xcb_xinerama_screen_info_t * xinerama_query_screens(xcb_connection_t *c, int *screen_number) { xcb_xinerama_query_screens_reply_t *reply; xcb_xinerama_screen_info_t *si; reply = xcb_xinerama_query_screens_reply(c, xcb_xinerama_query_screens_unchecked(c), NULL); if(!reply) { *screen_number = 0; return NULL; } *screen_number = xcb_xinerama_query_screens_screen_info_length(reply); si = xcb_xinerama_query_screens_screen_info(reply); p_delete(&reply); return si; } ScreensInfo * screensinfo_new(xcb_connection_t *conn) { ScreensInfo *si; xcb_xinerama_screen_info_t *xsi; int xinerama_screen_number, screen, screen_to_test; xcb_screen_t *s; bool drop; si = p_new(ScreensInfo, 1); if((si->xinerama_is_active = xinerama_is_active(conn))) { xsi = xinerama_query_screens(conn, &xinerama_screen_number); si->geometry = p_new(area_t, xinerama_screen_number); si->nscreen = 0; /* now check if screens overlaps (same x,y): if so, we take only the biggest one */ for(screen = 0; screen < xinerama_screen_number; screen++) { drop = false; for(screen_to_test = 0; screen_to_test < si->nscreen; screen_to_test++) if(xsi[screen].x_org == si->geometry[screen_to_test].x && xsi[screen].y_org == si->geometry[screen_to_test].y) { /* we already have a screen for this area, just check if * it's not bigger and drop it */ drop = true; si->geometry[screen_to_test].width = MAX(xsi[screen].width, xsi[screen_to_test].width); si->geometry[screen_to_test].height = MAX(xsi[screen].height, xsi[screen_to_test].height); } if(!drop) si->geometry[si->nscreen++] = screen_xsitoarea(xsi[screen]); } /* realloc smaller if xinerama_screen_number != screen registered */ if(xinerama_screen_number != si->nscreen) { area_t *newgeometry = p_new(area_t, si->nscreen); memcpy(newgeometry, si->geometry, si->nscreen * sizeof(area_t)); p_delete(&si->geometry); si->geometry = newgeometry; } p_delete(&xsi); } else { si->nscreen = xcb_setup_roots_length(xcb_get_setup(conn)); si->geometry = p_new(area_t, si->nscreen); for(screen = 0; screen < si->nscreen; screen++) { s = xcb_aux_get_screen(conn, screen); si->geometry[screen].x = 0; si->geometry[screen].y = 0; si->geometry[screen].width = s->width_in_pixels; si->geometry[screen].height = s->height_in_pixels; } } return si; } // vim: filetype=c:expandtab:shiftwidth=4:tabstop=8:softtabstop=4:encoding=utf-8:textwidth=80