2007-04-27 21:58:48 +04:00
|
|
|
/*
|
|
|
|
* calmwm - the calm window manager
|
|
|
|
*
|
|
|
|
* Copyright (c) 2004 Marius Aamodt Eriksen <marius@monkey.org>
|
2008-01-11 19:06:44 +03: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.
|
2007-04-27 21:58:48 +04:00
|
|
|
*
|
2011-05-11 17:53:51 +04:00
|
|
|
* $OpenBSD$
|
2007-04-27 21:58:48 +04:00
|
|
|
*/
|
|
|
|
|
2009-12-15 07:10:42 +03:00
|
|
|
#include <sys/param.h>
|
|
|
|
#include <sys/queue.h>
|
|
|
|
|
|
|
|
#include <err.h>
|
|
|
|
#include <errno.h>
|
2012-11-09 07:52:02 +04:00
|
|
|
#include <stdio.h>
|
2009-12-15 07:10:42 +03:00
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
|
2007-04-27 21:58:48 +04:00
|
|
|
#include "calmwm.h"
|
|
|
|
|
2012-11-29 07:54:46 +04:00
|
|
|
void
|
2013-04-29 04:56:47 +04:00
|
|
|
screen_init(int which)
|
2012-11-29 07:54:46 +04:00
|
|
|
{
|
2013-04-12 18:46:30 +04:00
|
|
|
struct screen_ctx *sc;
|
2012-11-29 07:54:46 +04:00
|
|
|
Window *wins, w0, w1;
|
|
|
|
XSetWindowAttributes rootattr;
|
2014-01-03 19:29:06 +04:00
|
|
|
unsigned int nwins, i;
|
2012-11-29 07:54:46 +04:00
|
|
|
|
2013-04-12 18:46:30 +04:00
|
|
|
sc = xcalloc(1, sizeof(*sc));
|
|
|
|
|
2014-01-27 19:13:09 +04:00
|
|
|
TAILQ_INIT(&sc->mruq);
|
2014-02-04 01:07:47 +04:00
|
|
|
TAILQ_INIT(&sc->regionq);
|
2014-01-27 19:13:09 +04:00
|
|
|
|
2012-11-29 07:54:46 +04:00
|
|
|
sc->which = which;
|
|
|
|
sc->rootwin = RootWindow(X_Dpy, sc->which);
|
2014-01-27 19:13:09 +04:00
|
|
|
conf_screen(sc);
|
2012-11-29 07:54:46 +04:00
|
|
|
|
|
|
|
xu_ewmh_net_supported(sc);
|
|
|
|
xu_ewmh_net_supported_wm_check(sc);
|
|
|
|
|
2013-05-01 01:12:20 +04:00
|
|
|
screen_update_geometry(sc);
|
|
|
|
group_init(sc);
|
2012-11-29 07:54:46 +04:00
|
|
|
|
2013-06-17 21:11:10 +04:00
|
|
|
rootattr.cursor = Conf.cursor[CF_NORMAL];
|
2012-12-18 22:39:55 +04:00
|
|
|
rootattr.event_mask = SubstructureRedirectMask|SubstructureNotifyMask|
|
|
|
|
PropertyChangeMask|EnterWindowMask|LeaveWindowMask|
|
|
|
|
ColormapChangeMask|BUTTONMASK;
|
2012-11-29 07:54:46 +04:00
|
|
|
|
|
|
|
XChangeWindowAttributes(X_Dpy, sc->rootwin,
|
|
|
|
CWEventMask|CWCursor, &rootattr);
|
|
|
|
|
|
|
|
/* Deal with existing clients. */
|
2014-02-04 00:20:39 +04:00
|
|
|
if (XQueryTree(X_Dpy, sc->rootwin, &w0, &w1, &wins, &nwins)) {
|
|
|
|
for (i = 0; i < nwins; i++)
|
|
|
|
(void)client_init(wins[i], sc);
|
2012-11-29 07:54:46 +04:00
|
|
|
|
2014-02-04 00:20:39 +04:00
|
|
|
XFree(wins);
|
|
|
|
}
|
2012-11-29 07:54:46 +04:00
|
|
|
screen_updatestackingorder(sc);
|
2014-02-08 06:49:30 +04:00
|
|
|
group_set_state(sc);
|
2012-11-29 07:54:46 +04:00
|
|
|
|
|
|
|
if (HasRandr)
|
|
|
|
XRRSelectInput(X_Dpy, sc->rootwin, RRScreenChangeNotifyMask);
|
|
|
|
|
2013-04-12 18:46:30 +04:00
|
|
|
TAILQ_INSERT_TAIL(&Screenq, sc, entry);
|
|
|
|
|
2012-11-29 07:54:46 +04:00
|
|
|
XSync(X_Dpy, False);
|
|
|
|
}
|
|
|
|
|
2007-04-27 21:58:48 +04:00
|
|
|
struct screen_ctx *
|
|
|
|
screen_fromroot(Window rootwin)
|
|
|
|
{
|
2008-07-11 18:21:28 +04:00
|
|
|
struct screen_ctx *sc;
|
2007-04-27 21:58:48 +04:00
|
|
|
|
2007-05-28 22:34:27 +04:00
|
|
|
TAILQ_FOREACH(sc, &Screenq, entry)
|
2007-04-27 21:58:48 +04:00
|
|
|
if (sc->rootwin == rootwin)
|
|
|
|
return (sc);
|
|
|
|
|
|
|
|
/* XXX FAIL HERE */
|
2007-05-28 22:34:27 +04:00
|
|
|
return (TAILQ_FIRST(&Screenq));
|
2007-04-27 21:58:48 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2009-12-10 20:16:51 +03:00
|
|
|
screen_updatestackingorder(struct screen_ctx *sc)
|
2007-04-27 21:58:48 +04:00
|
|
|
{
|
2008-07-11 18:21:28 +04:00
|
|
|
Window *wins, w0, w1;
|
|
|
|
struct client_ctx *cc;
|
2014-01-03 19:29:06 +04:00
|
|
|
unsigned int nwins, i, s;
|
2008-07-11 18:21:28 +04:00
|
|
|
|
2014-02-04 00:29:05 +04: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;
|
|
|
|
|
|
|
|
cc->stackingorder = s++;
|
|
|
|
}
|
|
|
|
XFree(wins);
|
2007-04-27 21:58:48 +04:00
|
|
|
}
|
|
|
|
}
|
2008-09-30 03:16:46 +04:00
|
|
|
|
2009-01-15 03:32:35 +03:00
|
|
|
/*
|
|
|
|
* Find which xinerama screen the coordinates (x,y) is on.
|
|
|
|
*/
|
2013-01-02 20:26:34 +04:00
|
|
|
struct geom
|
2013-12-14 02:39:13 +04:00
|
|
|
screen_find_xinerama(struct screen_ctx *sc, int x, int y, int flags)
|
2009-01-15 03:32:35 +03:00
|
|
|
{
|
2014-02-04 01:07:47 +04:00
|
|
|
struct region_ctx *region;
|
|
|
|
struct geom geom = sc->work;
|
|
|
|
|
|
|
|
TAILQ_FOREACH(region, &sc->regionq, entry) {
|
|
|
|
if (x >= region->area.x && x < region->area.x+region->area.w &&
|
|
|
|
y >= region->area.y && y < region->area.y+region->area.h) {
|
|
|
|
geom = region->area;
|
2013-01-02 20:26:34 +04:00
|
|
|
break;
|
|
|
|
}
|
2009-01-15 03:32:35 +03:00
|
|
|
}
|
2013-12-14 02:39:13 +04:00
|
|
|
if (flags & CWM_GAP) {
|
|
|
|
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);
|
|
|
|
}
|
2013-01-02 20:26:34 +04:00
|
|
|
return (geom);
|
2009-01-15 03:32:35 +03:00
|
|
|
}
|
2009-12-11 02:14:58 +03:00
|
|
|
|
|
|
|
void
|
2012-07-05 21:35:13 +04:00
|
|
|
screen_update_geometry(struct screen_ctx *sc)
|
2009-12-11 02:14:58 +03:00
|
|
|
{
|
2013-01-02 22:11:23 +04:00
|
|
|
XineramaScreenInfo *info = NULL;
|
2014-02-04 01:07:47 +04:00
|
|
|
struct region_ctx *region;
|
|
|
|
int info_no = 0, i;
|
2013-01-02 22:11:23 +04:00
|
|
|
|
2012-07-13 18:18:04 +04: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);
|
|
|
|
|
|
|
|
sc->work.x = sc->view.x + sc->gap.left;
|
|
|
|
sc->work.y = sc->view.y + sc->gap.top;
|
|
|
|
sc->work.w = sc->view.w - (sc->gap.left + sc->gap.right);
|
|
|
|
sc->work.h = sc->view.h - (sc->gap.top + sc->gap.bottom);
|
|
|
|
|
2013-01-02 22:11:23 +04:00
|
|
|
/* RandR event may have a CTRC added or removed. */
|
|
|
|
if (XineramaIsActive(X_Dpy))
|
|
|
|
info = XineramaQueryScreens(X_Dpy, &info_no);
|
2014-02-04 01:07:47 +04:00
|
|
|
|
|
|
|
while ((region = TAILQ_FIRST(&sc->regionq)) != NULL) {
|
|
|
|
TAILQ_REMOVE(&sc->regionq, region, entry);
|
|
|
|
free(region);
|
|
|
|
}
|
|
|
|
for (i = 0; i < info_no; i++) {
|
|
|
|
region = xmalloc(sizeof(*region));
|
|
|
|
region->num = i;
|
|
|
|
region->area.x = info[i].x_org;
|
|
|
|
region->area.y = info[i].y_org;
|
|
|
|
region->area.w = info[i].width;
|
|
|
|
region->area.h = info[i].height;
|
|
|
|
TAILQ_INSERT_TAIL(&sc->regionq, region, entry);
|
|
|
|
}
|
|
|
|
if (info)
|
|
|
|
XFree(info);
|
2012-07-06 18:18:00 +04:00
|
|
|
|
2012-07-03 17:49:03 +04:00
|
|
|
xu_ewmh_net_desktop_geometry(sc);
|
|
|
|
xu_ewmh_net_workarea(sc);
|
2009-12-11 02:14:58 +03:00
|
|
|
}
|