2007-04-27 11:58:48 -06:00
|
|
|
/*
|
|
|
|
* calmwm - the calm window manager
|
|
|
|
*
|
|
|
|
* Copyright (c) 2004 Marius Aamodt Eriksen <marius@monkey.org>
|
|
|
|
*
|
2008-01-11 09:06:44 -07:00
|
|
|
* Permission to use, copy, modify, and distribute this software for any
|
|
|
|
* purpose with or without fee is hereby granted, provided that the above
|
|
|
|
* copyright notice and this permission notice appear in all copies.
|
|
|
|
*
|
|
|
|
* THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
|
|
|
|
* WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
|
|
|
|
* MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
|
|
|
|
* ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
|
|
|
|
* WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
|
|
|
|
* ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
|
|
|
|
* OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
|
|
|
|
*
|
2018-01-23 09:18:59 -07:00
|
|
|
* $OpenBSD: screen.c,v 1.85 2018/01/23 16:18:59 okan Exp $
|
2007-04-27 11:58:48 -06:00
|
|
|
*/
|
|
|
|
|
2015-01-19 07:54:16 -07:00
|
|
|
#include <sys/types.h>
|
2009-12-14 21:10:42 -07:00
|
|
|
#include <sys/queue.h>
|
|
|
|
|
|
|
|
#include <err.h>
|
|
|
|
#include <errno.h>
|
2015-01-19 07:54:16 -07:00
|
|
|
#include <limits.h>
|
2012-11-08 20:52:02 -07:00
|
|
|
#include <stdio.h>
|
2009-12-14 21:10:42 -07:00
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
|
2007-04-27 11:58:48 -06:00
|
|
|
#include "calmwm.h"
|
|
|
|
|
2012-11-28 20:54:46 -07:00
|
|
|
void
|
2013-04-28 18:56:47 -06:00
|
|
|
screen_init(int which)
|
2012-11-28 20:54:46 -07:00
|
|
|
{
|
2013-04-12 08:46:30 -06:00
|
|
|
struct screen_ctx *sc;
|
2016-09-16 08:32:02 -06:00
|
|
|
Window *wins, w0, w1, active = None;
|
2012-11-28 20:54:46 -07:00
|
|
|
XSetWindowAttributes rootattr;
|
2018-01-23 09:18:59 -07:00
|
|
|
unsigned int nwins, w;
|
|
|
|
int i;
|
2012-11-28 20:54:46 -07:00
|
|
|
|
2014-10-08 09:31:01 -06:00
|
|
|
sc = xmalloc(sizeof(*sc));
|
2013-04-12 08:46:30 -06:00
|
|
|
|
2014-09-08 14:11:22 -06:00
|
|
|
TAILQ_INIT(&sc->clientq);
|
2014-02-03 14:07:47 -07:00
|
|
|
TAILQ_INIT(&sc->regionq);
|
2014-09-23 07:45:48 -06:00
|
|
|
TAILQ_INIT(&sc->groupq);
|
2014-01-27 08:13:09 -07:00
|
|
|
|
2012-11-28 20:54:46 -07:00
|
|
|
sc->which = which;
|
|
|
|
sc->rootwin = RootWindow(X_Dpy, sc->which);
|
2014-09-23 07:45:48 -06:00
|
|
|
sc->cycling = 0;
|
|
|
|
sc->hideall = 0;
|
2014-10-08 09:31:01 -06:00
|
|
|
|
2014-01-27 08:13:09 -07:00
|
|
|
conf_screen(sc);
|
2012-11-28 20:54:46 -07:00
|
|
|
|
|
|
|
xu_ewmh_net_supported(sc);
|
|
|
|
xu_ewmh_net_supported_wm_check(sc);
|
|
|
|
|
2013-04-30 15:12:20 -06:00
|
|
|
screen_update_geometry(sc);
|
2014-10-08 06:48:51 -06:00
|
|
|
|
2016-10-04 09:18:20 -06:00
|
|
|
for (i = 0; i < Conf.ngroups; i++)
|
2014-10-08 06:48:51 -06:00
|
|
|
group_init(sc, i);
|
2012-11-28 20:54:46 -07:00
|
|
|
|
2014-09-23 07:45:48 -06:00
|
|
|
xu_ewmh_net_desktop_names(sc);
|
|
|
|
xu_ewmh_net_wm_desktop_viewport(sc);
|
|
|
|
xu_ewmh_net_wm_number_of_desktops(sc);
|
|
|
|
xu_ewmh_net_showing_desktop(sc);
|
|
|
|
xu_ewmh_net_virtual_roots(sc);
|
2016-09-16 08:32:02 -06:00
|
|
|
active = xu_ewmh_get_net_active_window(sc);
|
2014-09-23 07:45:48 -06:00
|
|
|
|
2013-06-17 11:11:10 -06:00
|
|
|
rootattr.cursor = Conf.cursor[CF_NORMAL];
|
2015-08-21 10:52:37 -06:00
|
|
|
rootattr.event_mask = SubstructureRedirectMask |
|
|
|
|
SubstructureNotifyMask | PropertyChangeMask | EnterWindowMask |
|
|
|
|
LeaveWindowMask | ColormapChangeMask | BUTTONMASK;
|
2012-11-28 20:54:46 -07:00
|
|
|
|
|
|
|
XChangeWindowAttributes(X_Dpy, sc->rootwin,
|
2015-08-21 10:52:37 -06:00
|
|
|
(CWEventMask | CWCursor), &rootattr);
|
2012-11-28 20:54:46 -07:00
|
|
|
|
|
|
|
/* Deal with existing clients. */
|
2014-02-03 13:20:39 -07:00
|
|
|
if (XQueryTree(X_Dpy, sc->rootwin, &w0, &w1, &wins, &nwins)) {
|
2018-01-23 09:18:59 -07:00
|
|
|
for (w = 0; w < nwins; w++)
|
|
|
|
(void)client_init(wins[w], sc, (active == wins[w]));
|
2012-11-28 20:54:46 -07:00
|
|
|
|
2014-02-03 13:20:39 -07:00
|
|
|
XFree(wins);
|
|
|
|
}
|
2012-11-28 20:54:46 -07:00
|
|
|
screen_updatestackingorder(sc);
|
|
|
|
|
2016-10-18 11:03:30 -06:00
|
|
|
if (Conf.xrandr)
|
2012-11-28 20:54:46 -07:00
|
|
|
XRRSelectInput(X_Dpy, sc->rootwin, RRScreenChangeNotifyMask);
|
|
|
|
|
2013-04-12 08:46:30 -06:00
|
|
|
TAILQ_INSERT_TAIL(&Screenq, sc, entry);
|
|
|
|
|
2012-11-28 20:54:46 -07:00
|
|
|
XSync(X_Dpy, False);
|
|
|
|
}
|
|
|
|
|
2007-04-27 11:58:48 -06:00
|
|
|
struct screen_ctx *
|
2014-09-07 11:38:38 -06:00
|
|
|
screen_find(Window win)
|
2007-04-27 11:58:48 -06:00
|
|
|
{
|
2008-07-11 08:21:28 -06:00
|
|
|
struct screen_ctx *sc;
|
2007-04-27 11:58:48 -06:00
|
|
|
|
2014-09-07 13:27:30 -06:00
|
|
|
TAILQ_FOREACH(sc, &Screenq, entry) {
|
2014-09-07 11:38:38 -06:00
|
|
|
if (sc->rootwin == win)
|
2014-09-07 13:27:30 -06:00
|
|
|
return(sc);
|
|
|
|
}
|
2016-10-24 11:39:38 -06:00
|
|
|
warnx("%s: failure win 0x%lu\n", __func__, win);
|
2015-06-30 08:01:43 -06:00
|
|
|
return(NULL);
|
2007-04-27 11:58:48 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2009-12-10 10:16:51 -07:00
|
|
|
screen_updatestackingorder(struct screen_ctx *sc)
|
2007-04-27 11:58:48 -06:00
|
|
|
{
|
2008-07-11 08:21:28 -06:00
|
|
|
Window *wins, w0, w1;
|
|
|
|
struct client_ctx *cc;
|
2014-01-03 08:29:06 -07:00
|
|
|
unsigned int nwins, i, s;
|
2008-07-11 08:21:28 -06:00
|
|
|
|
2014-02-03 13:29:05 -07:00
|
|
|
if (XQueryTree(X_Dpy, sc->rootwin, &w0, &w1, &wins, &nwins)) {
|
|
|
|
for (s = 0, i = 0; i < nwins; i++) {
|
|
|
|
/* Skip hidden windows */
|
|
|
|
if ((cc = client_find(wins[i])) == NULL ||
|
|
|
|
cc->flags & CLIENT_HIDDEN)
|
|
|
|
continue;
|
2014-08-20 06:35:39 -06:00
|
|
|
|
2014-02-03 13:29:05 -07:00
|
|
|
cc->stackingorder = s++;
|
|
|
|
}
|
|
|
|
XFree(wins);
|
2007-04-27 11:58:48 -06:00
|
|
|
}
|
|
|
|
}
|
2008-09-29 17:16:46 -06:00
|
|
|
|
2015-11-09 13:03:29 -07:00
|
|
|
struct region_ctx *
|
|
|
|
region_find(struct screen_ctx *sc, int x, int y)
|
2009-01-14 17:32:35 -07:00
|
|
|
{
|
2015-06-26 12:54:25 -06:00
|
|
|
struct region_ctx *rc;
|
2014-02-03 14:07:47 -07:00
|
|
|
|
2015-06-26 12:54:25 -06:00
|
|
|
TAILQ_FOREACH(rc, &sc->regionq, entry) {
|
2015-11-09 13:03:29 -07:00
|
|
|
if ((x >= rc->view.x) && (x < (rc->view.x + rc->view.w)) &&
|
|
|
|
(y >= rc->view.y) && (y < (rc->view.y + rc->view.h))) {
|
2013-01-02 09:26:34 -07:00
|
|
|
break;
|
|
|
|
}
|
2009-01-14 17:32:35 -07:00
|
|
|
}
|
2015-11-09 13:03:29 -07:00
|
|
|
return(rc);
|
2009-01-14 17:32:35 -07:00
|
|
|
}
|
2009-12-10 16:14:58 -07:00
|
|
|
|
2015-11-11 07:22:01 -07:00
|
|
|
struct geom
|
2016-10-18 11:03:30 -06:00
|
|
|
screen_area(struct screen_ctx *sc, int x, int y, enum apply_gap apply_gap)
|
2015-11-11 07:22:01 -07:00
|
|
|
{
|
|
|
|
struct region_ctx *rc;
|
|
|
|
struct geom area = sc->work;
|
|
|
|
|
|
|
|
TAILQ_FOREACH(rc, &sc->regionq, entry) {
|
|
|
|
if ((x >= rc->area.x) && (x < (rc->area.x + rc->area.w)) &&
|
|
|
|
(y >= rc->area.y) && (y < (rc->area.y + rc->area.h))) {
|
|
|
|
area = rc->area;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2016-10-18 11:03:30 -06:00
|
|
|
if (apply_gap)
|
2015-11-11 07:22:01 -07:00
|
|
|
area = screen_apply_gap(sc, area);
|
|
|
|
return(area);
|
|
|
|
}
|
|
|
|
|
2009-12-10 16:14:58 -07:00
|
|
|
void
|
2012-07-05 11:35:13 -06:00
|
|
|
screen_update_geometry(struct screen_ctx *sc)
|
2009-12-10 16:14:58 -07:00
|
|
|
{
|
2015-06-26 12:54:25 -06:00
|
|
|
struct region_ctx *rc;
|
2013-01-02 11:11:23 -07:00
|
|
|
|
2012-07-13 08:18:04 -06:00
|
|
|
sc->view.x = 0;
|
|
|
|
sc->view.y = 0;
|
|
|
|
sc->view.w = DisplayWidth(X_Dpy, sc->which);
|
|
|
|
sc->view.h = DisplayHeight(X_Dpy, sc->which);
|
2015-06-26 10:11:21 -06:00
|
|
|
sc->work = screen_apply_gap(sc, sc->view);
|
2012-07-13 08:18:04 -06:00
|
|
|
|
2015-06-26 12:54:25 -06:00
|
|
|
while ((rc = TAILQ_FIRST(&sc->regionq)) != NULL) {
|
|
|
|
TAILQ_REMOVE(&sc->regionq, rc, entry);
|
|
|
|
free(rc);
|
2014-02-03 14:07:47 -07:00
|
|
|
}
|
2015-06-26 09:21:58 -06:00
|
|
|
|
2016-10-18 11:03:30 -06:00
|
|
|
if (Conf.xrandr) {
|
2015-06-26 09:21:58 -06:00
|
|
|
XRRScreenResources *sr;
|
|
|
|
XRRCrtcInfo *ci;
|
2015-11-09 13:03:29 -07:00
|
|
|
int i;
|
2015-06-26 09:21:58 -06:00
|
|
|
|
|
|
|
sr = XRRGetScreenResources(X_Dpy, sc->rootwin);
|
|
|
|
for (i = 0, ci = NULL; i < sr->ncrtc; i++) {
|
|
|
|
ci = XRRGetCrtcInfo(X_Dpy, sr, sr->crtcs[i]);
|
|
|
|
if (ci == NULL)
|
|
|
|
continue;
|
|
|
|
if (ci->noutput == 0) {
|
|
|
|
XRRFreeCrtcInfo(ci);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2015-06-26 12:54:25 -06:00
|
|
|
rc = xmalloc(sizeof(*rc));
|
|
|
|
rc->num = i;
|
2015-11-11 07:22:01 -07:00
|
|
|
rc->area.x = ci->x;
|
|
|
|
rc->area.y = ci->y;
|
|
|
|
rc->area.w = ci->width;
|
|
|
|
rc->area.h = ci->height;
|
2015-11-09 13:03:29 -07:00
|
|
|
rc->view.x = ci->x;
|
|
|
|
rc->view.y = ci->y;
|
|
|
|
rc->view.w = ci->width;
|
|
|
|
rc->view.h = ci->height;
|
|
|
|
rc->work = screen_apply_gap(sc, rc->view);
|
2015-06-26 12:54:25 -06:00
|
|
|
TAILQ_INSERT_TAIL(&sc->regionq, rc, entry);
|
2015-06-26 09:21:58 -06:00
|
|
|
|
|
|
|
XRRFreeCrtcInfo(ci);
|
|
|
|
}
|
|
|
|
XRRFreeScreenResources(sr);
|
2015-11-09 13:03:29 -07:00
|
|
|
} else {
|
|
|
|
rc = xmalloc(sizeof(*rc));
|
|
|
|
rc->num = 0;
|
|
|
|
rc->view.x = 0;
|
|
|
|
rc->view.y = 0;
|
|
|
|
rc->view.w = DisplayWidth(X_Dpy, sc->which);
|
|
|
|
rc->view.h = DisplayHeight(X_Dpy, sc->which);
|
|
|
|
rc->work = screen_apply_gap(sc, rc->view);
|
|
|
|
TAILQ_INSERT_TAIL(&sc->regionq, rc, entry);
|
2014-02-03 14:07:47 -07:00
|
|
|
}
|
2012-07-06 08:18:00 -06:00
|
|
|
|
2012-07-03 07:49:03 -06:00
|
|
|
xu_ewmh_net_desktop_geometry(sc);
|
|
|
|
xu_ewmh_net_workarea(sc);
|
2009-12-10 16:14:58 -07:00
|
|
|
}
|
2015-06-26 10:11:21 -06:00
|
|
|
|
|
|
|
struct geom
|
|
|
|
screen_apply_gap(struct screen_ctx *sc, struct geom geom)
|
|
|
|
{
|
|
|
|
geom.x += sc->gap.left;
|
|
|
|
geom.y += sc->gap.top;
|
|
|
|
geom.w -= (sc->gap.left + sc->gap.right);
|
|
|
|
geom.h -= (sc->gap.top + sc->gap.bottom);
|
|
|
|
|
|
|
|
return(geom);
|
|
|
|
}
|
2016-10-04 14:15:55 -06:00
|
|
|
|
|
|
|
/* Bring back clients which are beyond the screen. */
|
|
|
|
void
|
|
|
|
screen_assert_clients_within(struct screen_ctx *sc)
|
|
|
|
{
|
|
|
|
struct client_ctx *cc;
|
|
|
|
int top, left, right, bottom;
|
|
|
|
|
|
|
|
TAILQ_FOREACH(cc, &sc->clientq, entry) {
|
|
|
|
if (cc->sc != sc)
|
|
|
|
continue;
|
|
|
|
top = cc->geom.y;
|
|
|
|
left = cc->geom.x;
|
|
|
|
right = cc->geom.x + cc->geom.w + (cc->bwidth * 2) - 1;
|
|
|
|
bottom = cc->geom.y + cc->geom.h + (cc->bwidth * 2) - 1;
|
|
|
|
if ((top > sc->view.h || left > sc->view.w) ||
|
|
|
|
(bottom < 0 || right < 0)) {
|
|
|
|
cc->geom.x = sc->gap.left;
|
|
|
|
cc->geom.y = sc->gap.top;
|
|
|
|
client_move(cc);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|