1999-10-04 15:06:00 -07:00
|
|
|
/*
|
2007-01-13 11:14:29 -08:00
|
|
|
* Copyright (C) 2000-2007 Carsten Haitzler, Geoff Harrison and various contributors
|
2021-08-22 00:37:31 -07:00
|
|
|
* Copyright (C) 2004-2021 Kim Woelders
|
2000-04-05 16:22:56 -07:00
|
|
|
*
|
1999-10-04 15:06:00 -07:00
|
|
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
|
|
* of this software and associated documentation files (the "Software"), to
|
|
|
|
* deal in the Software without restriction, including without limitation the
|
|
|
|
* rights to use, copy, modify, merge, publish, distribute, sublicense, and/or
|
|
|
|
* sell copies of the Software, and to permit persons to whom the Software is
|
|
|
|
* furnished to do so, subject to the following conditions:
|
2000-04-05 16:22:56 -07:00
|
|
|
*
|
1999-10-04 15:06:00 -07:00
|
|
|
* The above copyright notice and this permission notice shall be included in
|
|
|
|
* all copies of the Software, its documentation and marketing & publicity
|
|
|
|
* materials, and acknowledgment shall be given in the documentation, materials
|
|
|
|
* and software packages that this Software was used.
|
2000-04-05 16:22:56 -07:00
|
|
|
*
|
1999-10-04 15:06:00 -07:00
|
|
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
|
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
|
|
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
|
|
|
|
* THE AUTHORS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER
|
|
|
|
* IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
|
|
|
|
* CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
|
|
|
*/
|
2015-01-15 22:25:38 -08:00
|
|
|
#include "config.h"
|
|
|
|
|
|
|
|
#include <X11/Xlib.h>
|
|
|
|
#include <X11/Xutil.h>
|
|
|
|
#if USE_XSYNC
|
|
|
|
#include <X11/extensions/sync.h>
|
|
|
|
#endif
|
|
|
|
|
1999-08-17 15:56:46 -07:00
|
|
|
#include "E.h"
|
2005-09-04 00:27:20 -07:00
|
|
|
#include "desktops.h"
|
2005-07-16 09:57:45 -07:00
|
|
|
#include "ewins.h"
|
2005-10-27 16:18:35 -07:00
|
|
|
#include "hints.h"
|
2006-04-19 12:13:46 -07:00
|
|
|
#include "session.h"
|
2013-04-07 04:06:58 -07:00
|
|
|
#include "xprop.h"
|
2005-07-16 09:57:45 -07:00
|
|
|
#include "xwin.h"
|
1999-08-17 15:56:46 -07:00
|
|
|
|
2023-10-21 02:04:12 -07:00
|
|
|
static void ICCCM_SetIconSizes(void);
|
2005-10-30 11:40:49 -08:00
|
|
|
|
2003-06-14 05:03:09 -07:00
|
|
|
void
|
|
|
|
ICCCM_Init(void)
|
|
|
|
{
|
2023-10-21 02:04:12 -07:00
|
|
|
ex_icccm_init();
|
2004-12-28 15:46:49 -08:00
|
|
|
|
2023-10-21 02:04:12 -07:00
|
|
|
Mode.current_cmap = WinGetCmap(VROOT);
|
2010-03-15 15:19:00 -07:00
|
|
|
|
2023-10-21 02:04:12 -07:00
|
|
|
ICCCM_SetIconSizes();
|
2004-12-28 15:46:49 -08:00
|
|
|
|
2023-10-21 02:04:12 -07:00
|
|
|
if (Mode.wm.window)
|
|
|
|
{
|
|
|
|
EX_Atom wm_props[1];
|
2005-10-30 11:40:49 -08:00
|
|
|
|
2023-10-21 02:04:12 -07:00
|
|
|
wm_props[0] = ea_i.WM_DELETE_WINDOW;
|
|
|
|
ex_window_prop_atom_set(WinGetXwin(VROOT), ea_i.WM_PROTOCOLS,
|
|
|
|
wm_props, 1);
|
|
|
|
}
|
2003-06-14 05:03:09 -07:00
|
|
|
}
|
|
|
|
|
2007-04-15 07:54:11 -07:00
|
|
|
int
|
2023-10-21 02:04:12 -07:00
|
|
|
ICCCM_ProcessClientClientMessage(EWin *ewin, XClientMessageEvent *event)
|
2003-06-14 05:03:09 -07:00
|
|
|
{
|
2023-10-21 02:04:12 -07:00
|
|
|
if (event->message_type == ea_i.WM_CHANGE_STATE)
|
|
|
|
{
|
|
|
|
if (event->data.l[0] == IconicState)
|
|
|
|
{
|
|
|
|
EwinIconify(ewin);
|
|
|
|
}
|
|
|
|
return 1;
|
|
|
|
}
|
2007-04-16 09:48:46 -07:00
|
|
|
|
2023-10-21 02:04:12 -07:00
|
|
|
return 0;
|
2007-04-16 09:48:46 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
int
|
2023-10-21 02:04:12 -07:00
|
|
|
ICCCM_ProcessRootClientMessage(XClientMessageEvent *event)
|
2007-04-16 09:48:46 -07:00
|
|
|
{
|
2023-10-21 02:04:12 -07:00
|
|
|
EX_Atom a;
|
2007-04-16 09:48:46 -07:00
|
|
|
|
2023-10-21 02:04:12 -07:00
|
|
|
if (event->message_type == ea_i.WM_PROTOCOLS)
|
|
|
|
{
|
|
|
|
a = event->data.l[0];
|
|
|
|
if (a == ea_i.WM_DELETE_WINDOW)
|
|
|
|
SessionExit(EEXIT_EXIT, NULL);
|
|
|
|
return 1;
|
|
|
|
}
|
2007-04-15 07:54:11 -07:00
|
|
|
|
2023-10-21 02:04:12 -07:00
|
|
|
return 0;
|
2003-06-14 05:03:09 -07:00
|
|
|
}
|
|
|
|
|
2004-12-28 15:46:49 -08:00
|
|
|
void
|
2023-10-21 02:04:12 -07:00
|
|
|
ICCCM_GetTitle(EWin *ewin)
|
2004-12-28 15:46:49 -08:00
|
|
|
{
|
2023-10-21 02:04:12 -07:00
|
|
|
EFREE_NULL(EwinGetIcccmName(ewin));
|
2004-12-28 15:46:49 -08:00
|
|
|
|
2023-10-21 02:04:12 -07:00
|
|
|
EwinGetIcccmName(ewin) = ex_icccm_title_get(EwinGetClientXwin(ewin));
|
2004-12-28 15:46:49 -08:00
|
|
|
|
2023-10-21 02:04:12 -07:00
|
|
|
EwinChange(ewin, EWIN_CHANGE_NAME);
|
1999-08-17 15:56:46 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2023-10-21 02:04:12 -07:00
|
|
|
ICCCM_Delete(const EWin *ewin)
|
1999-08-17 15:56:46 -07:00
|
|
|
{
|
2023-10-21 02:04:12 -07:00
|
|
|
if (EwinIsInternal(ewin))
|
|
|
|
{
|
|
|
|
EwinHide((EWin *) ewin);
|
|
|
|
return;
|
|
|
|
}
|
2003-12-18 15:43:26 -08:00
|
|
|
|
2023-10-21 02:04:12 -07:00
|
|
|
if (ewin->icccm.delete_window)
|
|
|
|
ex_icccm_delete_window_send(EwinGetClientXwin(ewin), CurrentTime);
|
|
|
|
else
|
|
|
|
XKillClient(disp, EwinGetClientXwin(ewin));
|
1999-08-17 15:56:46 -07:00
|
|
|
}
|
|
|
|
|
2023-10-21 02:04:12 -07:00
|
|
|
#if 0 /* Deprecated */
|
1999-08-17 15:56:46 -07:00
|
|
|
void
|
2023-10-21 02:04:12 -07:00
|
|
|
ICCCM_Save(const EWin *ewin)
|
1999-08-17 15:56:46 -07:00
|
|
|
{
|
2023-10-21 02:04:12 -07:00
|
|
|
if (EwinIsInternal(ewin))
|
|
|
|
return;
|
2003-12-18 15:43:26 -08:00
|
|
|
|
2023-10-21 02:04:12 -07:00
|
|
|
ex_icccm_send_save_yourself(EwinGetClientXwin(ewin));
|
1999-08-17 15:56:46 -07:00
|
|
|
}
|
2004-12-28 15:46:49 -08:00
|
|
|
#endif
|
1999-08-17 15:56:46 -07:00
|
|
|
|
|
|
|
void
|
2023-10-21 02:04:12 -07:00
|
|
|
ICCCM_Iconify(const EWin *ewin)
|
1999-08-17 15:56:46 -07:00
|
|
|
{
|
2023-10-21 02:04:12 -07:00
|
|
|
EUnmapWindow(EwinGetClientWin(ewin));
|
|
|
|
ex_icccm_state_set_iconic(EwinGetClientXwin(ewin));
|
1999-08-17 15:56:46 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2023-10-21 02:04:12 -07:00
|
|
|
ICCCM_DeIconify(const EWin *ewin)
|
1999-08-17 15:56:46 -07:00
|
|
|
{
|
2023-10-21 02:04:12 -07:00
|
|
|
EMapWindow(EwinGetClientWin(ewin));
|
|
|
|
ex_icccm_state_set_normal(EwinGetClientXwin(ewin));
|
2004-12-28 15:46:49 -08:00
|
|
|
}
|
2003-12-18 15:43:26 -08:00
|
|
|
|
2004-12-28 15:46:49 -08:00
|
|
|
void
|
2023-10-21 02:04:12 -07:00
|
|
|
ICCCM_Withdraw(const EWin *ewin)
|
2004-12-28 15:46:49 -08:00
|
|
|
{
|
2023-10-21 02:04:12 -07:00
|
|
|
/* We have a choice of deleting the WM_STATE property
|
|
|
|
* or changing the value to Withdrawn. Since twm/fvwm does
|
|
|
|
* it that way, we change it to Withdrawn.
|
|
|
|
*/
|
|
|
|
ex_icccm_state_set_withdrawn(EwinGetClientXwin(ewin));
|
2004-12-28 15:46:49 -08:00
|
|
|
|
2023-10-21 02:04:12 -07:00
|
|
|
XRemoveFromSaveSet(disp, EwinGetClientXwin(ewin));
|
1999-08-17 15:56:46 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2023-10-21 02:04:12 -07:00
|
|
|
ICCCM_SizeMatch(const EWin *ewin, int wi, int hi, int *pwo, int *pho)
|
|
|
|
{
|
|
|
|
int w, h;
|
|
|
|
int i, j;
|
|
|
|
float aspect, dw, dh;
|
|
|
|
|
|
|
|
w = wi;
|
|
|
|
h = hi;
|
|
|
|
|
|
|
|
if (w < ewin->icccm.width_min)
|
|
|
|
w = ewin->icccm.width_min;
|
|
|
|
if (w > ewin->icccm.width_max)
|
|
|
|
w = ewin->icccm.width_max;
|
|
|
|
if (h < ewin->icccm.height_min)
|
|
|
|
h = ewin->icccm.height_min;
|
|
|
|
if (h > ewin->icccm.height_max)
|
|
|
|
h = ewin->icccm.height_max;
|
|
|
|
|
|
|
|
if (w <= 0 || h <= 0)
|
|
|
|
return;
|
|
|
|
|
|
|
|
w -= ewin->icccm.base_w;
|
|
|
|
h -= ewin->icccm.base_h;
|
|
|
|
if ((w > 0) && (h > 0))
|
|
|
|
{
|
|
|
|
/* Ignore aspect ratio constraints when fullscreening */
|
|
|
|
if (!ewin->state.fullscreen)
|
|
|
|
{
|
|
|
|
aspect = ((float)w) / ((float)h);
|
|
|
|
dw = ewin->icccm.w_inc / 4.f;
|
|
|
|
dh = ewin->icccm.h_inc / 4.f;
|
|
|
|
if (Mode.mode == MODE_RESIZE_H)
|
|
|
|
{
|
|
|
|
if (aspect < ewin->icccm.aspect_min)
|
|
|
|
h = (int)((float)w / ewin->icccm.aspect_min + dh);
|
|
|
|
else if (aspect > ewin->icccm.aspect_max)
|
|
|
|
h = (int)((float)w / ewin->icccm.aspect_max + dh);
|
|
|
|
}
|
|
|
|
else if (Mode.mode == MODE_RESIZE_V)
|
|
|
|
{
|
|
|
|
if (aspect < ewin->icccm.aspect_min)
|
|
|
|
w = (int)((float)h * ewin->icccm.aspect_min + dw);
|
|
|
|
else if (aspect > ewin->icccm.aspect_max)
|
|
|
|
w = (int)((float)h * ewin->icccm.aspect_max + dw);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (aspect < ewin->icccm.aspect_min)
|
|
|
|
{
|
|
|
|
if (ewin->icccm.aspect_min >= 1.f)
|
|
|
|
h = (int)((float)w / ewin->icccm.aspect_min + dh);
|
|
|
|
else
|
|
|
|
w = (int)((float)h * ewin->icccm.aspect_min + dw);
|
|
|
|
}
|
|
|
|
else if (aspect > ewin->icccm.aspect_max)
|
|
|
|
{
|
|
|
|
if (ewin->icccm.aspect_max >= 1.f)
|
|
|
|
h = (int)((float)w / ewin->icccm.aspect_max + dh);
|
|
|
|
else
|
|
|
|
w = (int)((float)h * ewin->icccm.aspect_max + dw);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
i = w / ewin->icccm.w_inc;
|
|
|
|
j = h / ewin->icccm.h_inc;
|
|
|
|
w = i * ewin->icccm.w_inc;
|
|
|
|
h = j * ewin->icccm.h_inc;
|
|
|
|
}
|
|
|
|
w += ewin->icccm.base_w;
|
|
|
|
h += ewin->icccm.base_h;
|
|
|
|
|
|
|
|
*pwo = w;
|
|
|
|
*pho = h;
|
|
|
|
}
|
|
|
|
|
|
|
|
#if 0 /* Unused */
|
2005-01-30 03:02:43 -08:00
|
|
|
void
|
2023-10-21 02:04:12 -07:00
|
|
|
ICCCM_MatchSize(EWin *ewin)
|
2005-01-30 03:02:43 -08:00
|
|
|
{
|
2023-10-21 02:04:12 -07:00
|
|
|
ICCCM_SizeMatch(ewin, ewin->client.w, ewin->client.h, &ewin->client.w,
|
|
|
|
&ewin->client.h);
|
1999-08-17 15:56:46 -07:00
|
|
|
}
|
2007-09-05 12:12:56 -07:00
|
|
|
#endif
|
1999-08-17 15:56:46 -07:00
|
|
|
|
2005-10-12 10:20:42 -07:00
|
|
|
void
|
2023-10-21 02:04:12 -07:00
|
|
|
ICCCM_GetIncrementalSize(EWin *ewin, unsigned int w, unsigned int h,
|
|
|
|
unsigned int *wi, unsigned int *hi)
|
2005-10-12 10:20:42 -07:00
|
|
|
{
|
2023-10-21 02:04:12 -07:00
|
|
|
*wi = (w - ewin->icccm.base_w) / ewin->icccm.w_inc;
|
|
|
|
*hi = (h - ewin->icccm.base_h) / ewin->icccm.h_inc;
|
2005-10-12 10:20:42 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2023-10-21 02:04:12 -07:00
|
|
|
ICCCM_SetSizeConstraints(EWin *ewin, unsigned int wmin, unsigned int hmin,
|
|
|
|
unsigned int wmax, unsigned int hmax,
|
|
|
|
unsigned int wbase, unsigned int hbase,
|
|
|
|
unsigned int winc, unsigned int hinc,
|
|
|
|
float amin, float amax)
|
2005-10-12 10:20:42 -07:00
|
|
|
{
|
2023-10-21 02:04:12 -07:00
|
|
|
ewin->icccm.width_min = wmin;
|
|
|
|
ewin->icccm.height_min = hmin;
|
|
|
|
ewin->icccm.width_max = wmax;
|
|
|
|
ewin->icccm.height_max = hmax;
|
2005-10-12 10:20:42 -07:00
|
|
|
|
2023-10-21 02:04:12 -07:00
|
|
|
ewin->icccm.base_w = wbase;
|
|
|
|
ewin->icccm.base_h = hbase;
|
|
|
|
ewin->icccm.w_inc = winc;
|
|
|
|
ewin->icccm.h_inc = hinc;
|
2005-10-12 10:20:42 -07:00
|
|
|
|
2023-10-21 02:04:12 -07:00
|
|
|
ewin->icccm.aspect_min = amin;
|
|
|
|
ewin->icccm.aspect_max = amax;
|
2005-10-12 10:20:42 -07:00
|
|
|
|
2023-10-21 02:04:12 -07:00
|
|
|
ewin->props.no_resize_h = (wmin == wmax);
|
|
|
|
ewin->props.no_resize_v = (hmin == hmax);
|
2005-10-12 10:20:42 -07:00
|
|
|
}
|
|
|
|
|
1999-08-17 15:56:46 -07:00
|
|
|
void
|
2023-10-21 02:04:12 -07:00
|
|
|
ICCCM_Configure(EWin *ewin)
|
1999-08-17 15:56:46 -07:00
|
|
|
{
|
2023-10-21 02:04:12 -07:00
|
|
|
XEvent ev;
|
1999-08-17 15:56:46 -07:00
|
|
|
|
2023-10-21 02:04:12 -07:00
|
|
|
if (EwinIsInternal(ewin))
|
|
|
|
return;
|
2003-12-18 15:43:26 -08:00
|
|
|
|
2023-10-21 02:04:12 -07:00
|
|
|
ev.type = ConfigureNotify;
|
|
|
|
ev.xconfigure.display = disp;
|
|
|
|
ev.xconfigure.event = EwinGetClientXwin(ewin);
|
|
|
|
ev.xconfigure.window = EwinGetClientXwin(ewin);
|
|
|
|
#if 0 /* FIXME - Remove? */
|
|
|
|
Desk *dsk;
|
2005-10-18 11:00:53 -07:00
|
|
|
|
2023-10-21 02:04:12 -07:00
|
|
|
dsk = EoGetDesk(ewin);
|
|
|
|
ev.xconfigure.x = EoGetX(dsk) + ewin->client.x;
|
|
|
|
ev.xconfigure.y = EoGetY(dsk) + ewin->client.y;
|
2005-10-18 11:00:53 -07:00
|
|
|
#else
|
2023-10-21 02:04:12 -07:00
|
|
|
ev.xconfigure.x = ewin->client.x;
|
|
|
|
ev.xconfigure.y = ewin->client.y;
|
2005-10-18 11:00:53 -07:00
|
|
|
#endif
|
2023-10-21 02:04:12 -07:00
|
|
|
if (Mode.wm.window)
|
|
|
|
{
|
|
|
|
ev.xconfigure.x += Mode.wm.win_x;
|
|
|
|
ev.xconfigure.y += Mode.wm.win_y;
|
|
|
|
}
|
|
|
|
ev.xconfigure.width = ewin->client.w;
|
|
|
|
ev.xconfigure.height = ewin->client.h;
|
|
|
|
ev.xconfigure.border_width = 0;
|
|
|
|
ev.xconfigure.above = EoGetXwin(ewin);
|
|
|
|
ev.xconfigure.override_redirect = False;
|
|
|
|
EXSendEvent(EwinGetClientXwin(ewin), StructureNotifyMask, &ev);
|
1999-08-17 15:56:46 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2023-10-21 02:04:12 -07:00
|
|
|
ICCCM_AdoptStart(const EWin *ewin)
|
1999-08-17 15:56:46 -07:00
|
|
|
{
|
2023-10-21 02:04:12 -07:00
|
|
|
EX_Window win = EwinGetClientXwin(ewin);
|
2003-07-12 01:22:58 -07:00
|
|
|
|
2023-10-21 02:04:12 -07:00
|
|
|
if (!EwinIsInternal(ewin))
|
|
|
|
XAddToSaveSet(disp, win);
|
1999-08-17 15:56:46 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2023-10-21 02:04:12 -07:00
|
|
|
ICCCM_Adopt(const EWin *ewin)
|
1999-08-17 15:56:46 -07:00
|
|
|
{
|
2023-10-21 02:04:12 -07:00
|
|
|
EX_Window win = EwinGetClientXwin(ewin);
|
2003-12-18 15:43:26 -08:00
|
|
|
|
2023-10-21 02:04:12 -07:00
|
|
|
if (ewin->icccm.start_iconified)
|
|
|
|
ex_icccm_state_set_iconic(win);
|
|
|
|
else
|
|
|
|
ex_icccm_state_set_normal(win);
|
1999-08-17 15:56:46 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2023-10-21 02:04:12 -07:00
|
|
|
ICCCM_Cmap(EWin *ewin)
|
|
|
|
{
|
|
|
|
EX_Colormap ecmap, dcmap, ccmap;
|
|
|
|
XWindowAttributes xwa;
|
|
|
|
int i, num;
|
|
|
|
EX_Window *wlist;
|
|
|
|
|
|
|
|
ecmap = Mode.current_cmap;
|
|
|
|
dcmap = WinGetCmap(VROOT);
|
|
|
|
|
|
|
|
if (!ewin)
|
|
|
|
{
|
|
|
|
if (ecmap == dcmap)
|
|
|
|
return;
|
|
|
|
ccmap = dcmap;
|
|
|
|
goto set_cmap;
|
|
|
|
}
|
|
|
|
|
|
|
|
ccmap = EwinGetClientWin(ewin)->cmap;
|
|
|
|
|
|
|
|
if (ccmap == ecmap || EoGetWin(ewin)->argb)
|
|
|
|
return;
|
|
|
|
|
|
|
|
/* Hack - assume that if client cmap is default cmap it doesn't have
|
|
|
|
* WM_COLORMAP_WINDOWS */
|
|
|
|
if (ccmap == dcmap)
|
|
|
|
goto set_cmap;
|
|
|
|
|
|
|
|
num = ex_window_prop_window_list_get(EwinGetClientXwin(ewin),
|
|
|
|
ea_i.WM_COLORMAP_WINDOWS, &wlist);
|
|
|
|
if (num > 0)
|
|
|
|
{
|
|
|
|
for (i = 0; i < num; i++)
|
|
|
|
{
|
|
|
|
if (EXGetWindowAttributes(wlist[i], &xwa))
|
|
|
|
{
|
|
|
|
if (xwa.colormap != dcmap)
|
|
|
|
{
|
|
|
|
XInstallColormap(disp, xwa.colormap);
|
|
|
|
Mode.current_cmap = xwa.colormap;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
Efree(wlist);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
set_cmap:
|
|
|
|
if (EDebug(EDBUG_TYPE_FOCUS))
|
|
|
|
Eprintf("%s: %#x\n", __func__, ccmap);
|
|
|
|
XInstallColormap(disp, ccmap);
|
|
|
|
Mode.current_cmap = ccmap;
|
1999-08-17 15:56:46 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2023-10-21 02:04:12 -07:00
|
|
|
ICCCM_Focus(const EWin *ewin)
|
|
|
|
{
|
|
|
|
if (EDebug(EDBUG_TYPE_FOCUS))
|
|
|
|
{
|
|
|
|
if (ewin)
|
|
|
|
Eprintf("%s: T=%#x R=%#x %#x %s\n", __func__, Mode.events.time,
|
|
|
|
(int)NextRequest(disp), EwinGetClientXwin(ewin),
|
|
|
|
EwinGetTitle(ewin));
|
|
|
|
else
|
|
|
|
Eprintf("%s: T=%#x R=%#x None\n", __func__, Mode.events.time,
|
|
|
|
(int)NextRequest(disp));
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!ewin)
|
|
|
|
{
|
|
|
|
XSetInputFocus(disp, WinGetXwin(VROOT), RevertToPointerRoot,
|
|
|
|
Mode.events.time);
|
|
|
|
HintsSetActiveWindow(NoXID);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (ewin->icccm.take_focus)
|
|
|
|
{
|
|
|
|
ex_icccm_take_focus_send(EwinGetClientXwin(ewin), Mode.events.time);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (ewin->icccm.need_input)
|
|
|
|
XSetInputFocus(disp, EwinGetClientXwin(ewin),
|
|
|
|
RevertToPointerRoot, Mode.events.time);
|
|
|
|
|
|
|
|
HintsSetActiveWindow(EwinGetClientXwin(ewin));
|
1999-08-17 15:56:46 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2023-10-21 02:04:12 -07:00
|
|
|
ICCCM_GetGeoms(EWin *ewin)
|
|
|
|
{
|
|
|
|
XSizeHints hint;
|
|
|
|
long mask;
|
|
|
|
|
|
|
|
if (XGetWMNormalHints(disp, EwinGetClientXwin(ewin), &hint, &mask))
|
|
|
|
{
|
|
|
|
if (!(ewin->state.placed))
|
|
|
|
{
|
|
|
|
if ((hint.flags & USPosition) || ((hint.flags & PPosition)))
|
|
|
|
{
|
|
|
|
if ((hint.flags & PPosition) && (!EoIsSticky(ewin)))
|
|
|
|
{
|
|
|
|
Desk *dsk;
|
|
|
|
|
|
|
|
dsk = EoGetDesk(ewin);
|
|
|
|
if (!dsk)
|
|
|
|
dsk = DesksGetCurrent();
|
|
|
|
ewin->client.x -= EoGetX(dsk);
|
|
|
|
ewin->client.y -= EoGetY(dsk);
|
|
|
|
if (ewin->client.x + ewin->client.w >= WinGetW(VROOT))
|
|
|
|
{
|
|
|
|
ewin->client.x += EoGetX(dsk);
|
|
|
|
}
|
|
|
|
else if (ewin->client.x < 0)
|
|
|
|
{
|
|
|
|
ewin->client.x += EoGetX(dsk);
|
|
|
|
}
|
|
|
|
if (ewin->client.y + ewin->client.h >= WinGetH(VROOT))
|
|
|
|
{
|
|
|
|
ewin->client.y += EoGetY(dsk);
|
|
|
|
}
|
|
|
|
else if (ewin->client.y < 0)
|
|
|
|
{
|
|
|
|
ewin->client.y += EoGetY(dsk);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
ewin->state.placed = 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (hint.flags & PMinSize)
|
|
|
|
{
|
|
|
|
ewin->icccm.width_min = MAX(0, hint.min_width);
|
|
|
|
ewin->icccm.height_min = MAX(0, hint.min_height);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
ewin->icccm.width_min = 0;
|
|
|
|
ewin->icccm.height_min = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (hint.flags & PMaxSize)
|
|
|
|
{
|
|
|
|
ewin->icccm.width_max = MIN(hint.max_width, 65535);
|
|
|
|
ewin->icccm.height_max = MIN(hint.max_height, 65535);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
ewin->icccm.width_max = 65535;
|
|
|
|
ewin->icccm.height_max = 65535;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (hint.flags & PResizeInc)
|
|
|
|
{
|
|
|
|
ewin->icccm.w_inc = MAX(1, hint.width_inc);
|
|
|
|
ewin->icccm.h_inc = MAX(1, hint.height_inc);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
ewin->icccm.w_inc = 1;
|
|
|
|
ewin->icccm.h_inc = 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (hint.flags & PAspect)
|
|
|
|
{
|
|
|
|
if (hint.min_aspect.y > 0 && hint.min_aspect.x > 0)
|
|
|
|
{
|
|
|
|
ewin->icccm.aspect_min =
|
|
|
|
((float)hint.min_aspect.x) / ((float)hint.min_aspect.y);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
ewin->icccm.aspect_min = 0.f;
|
|
|
|
}
|
|
|
|
if (hint.max_aspect.y > 0 && hint.max_aspect.x > 0)
|
|
|
|
{
|
|
|
|
ewin->icccm.aspect_max =
|
|
|
|
((float)hint.max_aspect.x) / ((float)hint.max_aspect.y);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
ewin->icccm.aspect_max = 65535.f;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
ewin->icccm.aspect_min = 0.f;
|
|
|
|
ewin->icccm.aspect_max = 65535.f;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (hint.flags & PBaseSize)
|
|
|
|
{
|
|
|
|
ewin->icccm.base_w = hint.base_width;
|
|
|
|
ewin->icccm.base_h = hint.base_height;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
ewin->icccm.base_w = ewin->icccm.width_min;
|
|
|
|
ewin->icccm.base_h = ewin->icccm.height_min;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (ewin->icccm.width_min < ewin->icccm.base_w)
|
|
|
|
ewin->icccm.width_min = ewin->icccm.base_w;
|
|
|
|
if (ewin->icccm.height_min < ewin->icccm.base_h)
|
|
|
|
ewin->icccm.height_min = ewin->icccm.base_h;
|
|
|
|
|
|
|
|
if (ewin->icccm.width_max < ewin->icccm.base_w)
|
|
|
|
ewin->icccm.width_max = ewin->icccm.base_w;
|
|
|
|
if (ewin->icccm.height_max < ewin->icccm.base_h)
|
|
|
|
ewin->icccm.height_max = ewin->icccm.base_h;
|
|
|
|
|
|
|
|
if (hint.flags & PWinGravity)
|
|
|
|
ewin->icccm.grav = hint.win_gravity;
|
|
|
|
else
|
|
|
|
ewin->icccm.grav = NorthWestGravity;
|
|
|
|
}
|
|
|
|
|
|
|
|
ewin->props.no_resize_h = (ewin->icccm.width_min == ewin->icccm.width_max);
|
|
|
|
ewin->props.no_resize_v =
|
|
|
|
(ewin->icccm.height_min == ewin->icccm.height_max);
|
|
|
|
|
|
|
|
if (EDebug(EDBUG_TYPE_SNAPS))
|
|
|
|
Eprintf("Snap get icccm %#x: %4d+%4d %4dx%4d: %s\n",
|
|
|
|
EwinGetClientXwin(ewin), ewin->client.x, ewin->client.y,
|
|
|
|
ewin->client.w, ewin->client.h, EwinGetTitle(ewin));
|
1999-08-17 15:56:46 -07:00
|
|
|
}
|
|
|
|
|
2013-12-28 09:59:09 -08:00
|
|
|
#define TryGroup(e) (((e)->icccm.group != NoXID) && ((e)->icccm.group != EwinGetClientXwin(e)))
|
2005-03-06 03:40:56 -08:00
|
|
|
|
2007-04-16 09:36:10 -07:00
|
|
|
static void
|
2023-10-21 02:04:12 -07:00
|
|
|
ICCCM_GetWmClass(EWin *ewin)
|
1999-08-17 15:56:46 -07:00
|
|
|
{
|
2023-10-21 02:04:12 -07:00
|
|
|
EFREE_NULL(EwinGetIcccmCName(ewin));
|
|
|
|
EFREE_NULL(EwinGetIcccmClass(ewin));
|
2007-04-16 09:36:10 -07:00
|
|
|
|
2023-10-21 02:04:12 -07:00
|
|
|
ex_icccm_name_class_get(EwinGetClientXwin(ewin),
|
|
|
|
&EwinGetIcccmCName(ewin), &EwinGetIcccmClass(ewin));
|
|
|
|
if (!EwinGetIcccmCName(ewin) && TryGroup(ewin))
|
|
|
|
ex_icccm_name_class_get(ewin->icccm.group,
|
|
|
|
&EwinGetIcccmCName(ewin),
|
|
|
|
&EwinGetIcccmClass(ewin));
|
2007-04-16 09:36:10 -07:00
|
|
|
}
|
2005-03-12 07:22:08 -08:00
|
|
|
|
2007-04-16 09:36:10 -07:00
|
|
|
static void
|
2023-10-21 02:04:12 -07:00
|
|
|
ICCCM_GetWmCommand(EWin *ewin)
|
2007-04-16 09:36:10 -07:00
|
|
|
{
|
2023-10-21 02:04:12 -07:00
|
|
|
int argc;
|
|
|
|
char **argv, s[4096], *ss;
|
2005-03-12 03:09:33 -08:00
|
|
|
|
2023-10-21 02:04:12 -07:00
|
|
|
EFREE_NULL(ewin->icccm.wm_command);
|
2000-07-26 11:00:47 -07:00
|
|
|
|
2023-10-21 02:04:12 -07:00
|
|
|
argc = ex_window_prop_string_list_get(EwinGetClientXwin(ewin),
|
|
|
|
ea_i.WM_COMMAND, &argv);
|
|
|
|
if ((argc < 0) && TryGroup(ewin))
|
|
|
|
argc = ex_window_prop_string_list_get(ewin->icccm.group,
|
|
|
|
ea_i.WM_COMMAND, &argv);
|
2004-02-28 17:30:18 -08:00
|
|
|
|
2023-10-21 02:04:12 -07:00
|
|
|
ss = StrlistEncodeEscaped(s, sizeof(s), argv, argc);
|
|
|
|
ewin->icccm.wm_command = Estrdup(ss);
|
|
|
|
StrlistFree(argv, argc);
|
2007-04-16 09:36:10 -07:00
|
|
|
}
|
2000-07-26 11:00:47 -07:00
|
|
|
|
2007-04-16 09:36:10 -07:00
|
|
|
static void
|
2023-10-21 02:04:12 -07:00
|
|
|
ICCCM_GetWmClientMachine(EWin *ewin)
|
2007-04-16 09:36:10 -07:00
|
|
|
{
|
2023-10-21 02:04:12 -07:00
|
|
|
EFREE_NULL(ewin->icccm.wm_machine);
|
2007-04-16 09:36:10 -07:00
|
|
|
|
2023-10-21 02:04:12 -07:00
|
|
|
ewin->icccm.wm_machine =
|
|
|
|
ex_window_prop_string_get(EwinGetClientXwin(ewin),
|
|
|
|
ea_i.WM_CLIENT_MACHINE);
|
|
|
|
if (!ewin->icccm.wm_machine && TryGroup(ewin))
|
|
|
|
ewin->icccm.wm_machine =
|
|
|
|
ex_window_prop_string_get(ewin->icccm.group,
|
|
|
|
ea_i.WM_CLIENT_MACHINE);
|
2007-04-16 09:36:10 -07:00
|
|
|
}
|
2003-05-22 12:15:03 -07:00
|
|
|
|
2007-04-16 09:36:10 -07:00
|
|
|
static void
|
2023-10-21 02:04:12 -07:00
|
|
|
ICCCM_GetWmIconName(EWin *ewin)
|
2007-04-16 09:36:10 -07:00
|
|
|
{
|
2023-10-21 02:04:12 -07:00
|
|
|
EFREE_NULL(ewin->icccm.wm_icon_name);
|
2007-04-16 09:36:10 -07:00
|
|
|
|
2023-10-21 02:04:12 -07:00
|
|
|
ewin->icccm.wm_icon_name =
|
|
|
|
ex_window_prop_string_get(EwinGetClientXwin(ewin), ea_i.WM_ICON_NAME);
|
|
|
|
if (!ewin->icccm.wm_icon_name && TryGroup(ewin))
|
|
|
|
ewin->icccm.wm_icon_name =
|
|
|
|
ex_window_prop_string_get(ewin->icccm.group, ea_i.WM_ICON_NAME);
|
2007-04-16 09:36:10 -07:00
|
|
|
}
|
2000-07-26 11:00:47 -07:00
|
|
|
|
2007-04-16 09:36:10 -07:00
|
|
|
static void
|
2023-10-21 02:04:12 -07:00
|
|
|
ICCCM_GetWmWindowRole(EWin *ewin)
|
2007-04-16 09:36:10 -07:00
|
|
|
{
|
2023-10-21 02:04:12 -07:00
|
|
|
EFREE_NULL(ewin->icccm.wm_role);
|
|
|
|
ewin->icccm.wm_role =
|
|
|
|
ex_window_prop_string_get(EwinGetClientXwin(ewin), ea_i.WM_WINDOW_ROLE);
|
1999-08-17 15:56:46 -07:00
|
|
|
}
|
2000-02-28 10:08:25 -08:00
|
|
|
|
1999-08-17 15:56:46 -07:00
|
|
|
void
|
2023-10-21 02:04:12 -07:00
|
|
|
ICCCM_GetInfo(EWin *ewin)
|
2007-04-16 09:36:10 -07:00
|
|
|
{
|
2023-10-21 02:04:12 -07:00
|
|
|
ICCCM_GetWmClass(ewin);
|
|
|
|
ICCCM_GetWmCommand(ewin);
|
|
|
|
ICCCM_GetWmClientMachine(ewin);
|
|
|
|
ICCCM_GetWmIconName(ewin);
|
|
|
|
ICCCM_GetWmWindowRole(ewin);
|
2007-04-16 09:36:10 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2023-10-21 02:04:12 -07:00
|
|
|
ICCCM_GetWmHints(EWin *ewin)
|
|
|
|
{
|
|
|
|
XWMHints *hint;
|
|
|
|
|
|
|
|
hint = XGetWMHints(disp, EwinGetClientXwin(ewin));
|
|
|
|
if (!hint)
|
|
|
|
return;
|
|
|
|
|
|
|
|
/* I have to make sure the thing i'm docking is a dock app */
|
|
|
|
if ((hint->flags & StateHint) && (hint->initial_state == WithdrawnState))
|
|
|
|
{
|
|
|
|
if (hint->flags & (StateHint | IconWindowHint | IconPositionHint |
|
|
|
|
WindowGroupHint))
|
|
|
|
{
|
|
|
|
if ((hint->icon_x == 0) && (hint->icon_y == 0)
|
|
|
|
&& hint->window_group == EwinGetClientXwin(ewin))
|
|
|
|
ewin->state.docked = 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
ewin->icccm.need_input =
|
|
|
|
((hint->flags & InputHint) && (!hint->input)) ? 0 : 1;
|
|
|
|
|
|
|
|
ewin->icccm.start_iconified =
|
|
|
|
((hint->flags & StateHint) &&
|
|
|
|
(hint->initial_state == IconicState)) ? 1 : 0;
|
|
|
|
|
|
|
|
if (hint->flags & IconPixmapHint)
|
|
|
|
{
|
|
|
|
if (ewin->icccm.icon_pmap != hint->icon_pixmap)
|
|
|
|
{
|
|
|
|
ewin->icccm.icon_pmap = hint->icon_pixmap;
|
|
|
|
EwinChange(ewin, EWIN_CHANGE_ICON_PMAP);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
ewin->icccm.icon_pmap = NoXID;
|
|
|
|
}
|
|
|
|
|
|
|
|
ewin->icccm.icon_mask =
|
|
|
|
(hint->flags & IconMaskHint) ? hint->icon_mask : NoXID;
|
|
|
|
|
|
|
|
ewin->icccm.icon_win =
|
|
|
|
(hint->flags & IconWindowHint) ? hint->icon_window : NoXID;
|
|
|
|
|
|
|
|
ewin->icccm.group =
|
|
|
|
(hint->flags & WindowGroupHint) ? hint->window_group : NoXID;
|
|
|
|
|
|
|
|
if (hint->flags & XUrgencyHint)
|
|
|
|
{
|
|
|
|
if (!ewin->state.attention)
|
|
|
|
EwinChange(ewin, EWIN_CHANGE_ATTENTION);
|
|
|
|
ewin->icccm.urgency = 1;
|
|
|
|
ewin->state.attention = 1;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
ewin->icccm.urgency = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (ewin->icccm.group == EwinGetClientXwin(ewin))
|
|
|
|
{
|
|
|
|
ewin->icccm.is_group_leader = 1;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
ewin->icccm.is_group_leader = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
XFree(hint);
|
2007-04-16 09:36:10 -07:00
|
|
|
}
|
2005-02-12 08:16:29 -08:00
|
|
|
|
2007-04-16 09:36:10 -07:00
|
|
|
static void
|
2023-10-21 02:04:12 -07:00
|
|
|
ICCCM_GetWmProtocols(EWin *ewin)
|
|
|
|
{
|
|
|
|
EX_Atom prop[64]; /* More is unlikely */
|
|
|
|
int i, num;
|
|
|
|
|
|
|
|
num = ex_window_prop_atom_get(EwinGetClientXwin(ewin), ea_i.WM_PROTOCOLS,
|
|
|
|
prop, 64);
|
|
|
|
if (num < 0)
|
|
|
|
return;
|
|
|
|
|
|
|
|
ewin->icccm.take_focus = 0;
|
|
|
|
ewin->icccm.delete_window = 0;
|
|
|
|
for (i = 0; i < num; i++)
|
|
|
|
{
|
|
|
|
if (prop[i] == ea_i.WM_TAKE_FOCUS)
|
|
|
|
ewin->icccm.take_focus = 1;
|
|
|
|
else if (prop[i] == ea_i.WM_DELETE_WINDOW)
|
|
|
|
ewin->icccm.delete_window = 1;
|
2007-04-16 09:36:10 -07:00
|
|
|
#if USE_XSYNC
|
2023-10-21 02:04:12 -07:00
|
|
|
else if (prop[i] == ea_n._NET_WM_SYNC_REQUEST)
|
|
|
|
{
|
|
|
|
unsigned int c;
|
|
|
|
|
|
|
|
ewin->ewmh.sync_request_enable = 1;
|
|
|
|
ex_window_prop_card32_get(EwinGetClientXwin(ewin),
|
|
|
|
ea_n._NET_WM_SYNC_REQUEST_COUNTER, &c, 1);
|
|
|
|
ewin->ewmh.sync_request_counter = c;
|
|
|
|
}
|
2013-12-27 07:31:18 -08:00
|
|
|
#endif
|
2023-10-21 02:04:12 -07:00
|
|
|
}
|
2007-04-16 09:36:10 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2023-10-21 02:04:12 -07:00
|
|
|
ICCCM_GetWmTransientFor(EWin *ewin)
|
2007-04-16 09:36:10 -07:00
|
|
|
{
|
2023-10-21 02:04:12 -07:00
|
|
|
int num;
|
|
|
|
EX_Window win;
|
2007-04-16 09:36:10 -07:00
|
|
|
|
2023-10-21 02:04:12 -07:00
|
|
|
num = ex_window_prop_window_get(EwinGetClientXwin(ewin),
|
|
|
|
ea_i.WM_TRANSIENT_FOR, &win, 1);
|
|
|
|
if (num > 0)
|
|
|
|
{
|
|
|
|
ewin->icccm.transient = 1;
|
|
|
|
ewin->icccm.transient_for = win;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
ewin->icccm.transient = 0;
|
|
|
|
ewin->icccm.transient_for = NoXID;
|
|
|
|
}
|
2007-04-16 09:36:10 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2023-10-21 02:04:12 -07:00
|
|
|
ICCCM_GetWmClientLeader(EWin *ewin)
|
2007-04-16 09:36:10 -07:00
|
|
|
{
|
2023-10-21 02:04:12 -07:00
|
|
|
int num;
|
|
|
|
EX_Window cleader;
|
2007-04-16 09:36:10 -07:00
|
|
|
|
2023-10-21 02:04:12 -07:00
|
|
|
num = ex_window_prop_window_get(EwinGetClientXwin(ewin),
|
|
|
|
ea_i.WM_CLIENT_LEADER, &cleader, 1);
|
|
|
|
if (num > 0)
|
|
|
|
{
|
|
|
|
ewin->icccm.client_leader = cleader;
|
|
|
|
if (!ewin->icccm.group)
|
|
|
|
ewin->icccm.group = cleader;
|
|
|
|
}
|
1999-08-17 15:56:46 -07:00
|
|
|
}
|
|
|
|
|
2007-04-16 09:36:10 -07:00
|
|
|
void
|
2023-10-21 02:04:12 -07:00
|
|
|
ICCCM_GetHints(EWin *ewin)
|
2007-04-16 09:36:10 -07:00
|
|
|
{
|
2023-10-21 02:04:12 -07:00
|
|
|
ICCCM_GetWmHints(ewin);
|
|
|
|
ICCCM_GetWmProtocols(ewin);
|
|
|
|
ICCCM_GetWmTransientFor(ewin);
|
|
|
|
ICCCM_GetWmClientLeader(ewin);
|
2007-04-16 09:36:10 -07:00
|
|
|
}
|
|
|
|
|
2012-10-24 10:20:38 -07:00
|
|
|
static void
|
2005-01-22 03:32:58 -08:00
|
|
|
ICCCM_SetIconSizes(void)
|
1999-08-17 15:56:46 -07:00
|
|
|
{
|
2023-10-21 02:04:12 -07:00
|
|
|
XIconSize *is;
|
1999-08-17 15:56:46 -07:00
|
|
|
|
2023-10-21 02:04:12 -07:00
|
|
|
is = XAllocIconSize();
|
|
|
|
is->min_width = 8;
|
|
|
|
is->min_height = 8;
|
|
|
|
is->max_width = 48;
|
|
|
|
is->max_height = 48;
|
|
|
|
is->width_inc = 1;
|
|
|
|
is->height_inc = 1;
|
|
|
|
XSetIconSizes(disp, WinGetXwin(VROOT), is, 1);
|
|
|
|
XFree(is);
|
1999-08-17 15:56:46 -07:00
|
|
|
}
|
|
|
|
|
2004-07-04 01:47:28 -07:00
|
|
|
/*
|
|
|
|
* Process received window property change
|
|
|
|
*/
|
2007-04-16 09:36:10 -07:00
|
|
|
int
|
2023-10-21 02:04:12 -07:00
|
|
|
ICCCM_ProcessPropertyChange(EWin *ewin, EX_Atom atom_change)
|
|
|
|
{
|
|
|
|
if (atom_change == ea_i.WM_NAME)
|
|
|
|
{
|
|
|
|
ICCCM_GetTitle(ewin);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* ICCCM_GetHints */
|
|
|
|
if (atom_change == ea_i.WM_HINTS)
|
|
|
|
{
|
|
|
|
ICCCM_GetWmHints(ewin);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
if (atom_change == ea_i.WM_PROTOCOLS)
|
|
|
|
{
|
|
|
|
ICCCM_GetWmProtocols(ewin);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
if (atom_change == ea_i.WM_TRANSIENT_FOR)
|
|
|
|
{
|
|
|
|
ICCCM_GetWmTransientFor(ewin);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
if (atom_change == ea_i.WM_CLIENT_LEADER)
|
|
|
|
{
|
|
|
|
ICCCM_GetWmClientLeader(ewin);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* ICCCM_GetInfo */
|
|
|
|
if (atom_change == ea_i.WM_ICON_NAME)
|
|
|
|
{
|
|
|
|
ICCCM_GetWmIconName(ewin);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
#if 1 /* FIXME - Any reason to process these? */
|
|
|
|
if (atom_change == ea_i.WM_CLASS)
|
|
|
|
{
|
|
|
|
ICCCM_GetWmClass(ewin);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
if (atom_change == ea_i.WM_COMMAND)
|
|
|
|
{
|
|
|
|
ICCCM_GetWmCommand(ewin);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
if (atom_change == ea_i.WM_CLIENT_MACHINE)
|
|
|
|
{
|
|
|
|
ICCCM_GetWmClientMachine(ewin);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
if (atom_change == ea_i.WM_WINDOW_ROLE)
|
|
|
|
{
|
|
|
|
ICCCM_GetWmWindowRole(ewin);
|
|
|
|
return 1;
|
|
|
|
}
|
2007-04-16 09:36:10 -07:00
|
|
|
#endif
|
|
|
|
|
2023-10-21 02:04:12 -07:00
|
|
|
if (atom_change == ea_i.WM_COLORMAP_WINDOWS)
|
|
|
|
{
|
|
|
|
ICCCM_Cmap(ewin);
|
|
|
|
return 1;
|
|
|
|
}
|
2007-04-16 09:36:10 -07:00
|
|
|
|
2023-10-21 02:04:12 -07:00
|
|
|
if (atom_change == ea_i.WM_NORMAL_HINTS)
|
|
|
|
{
|
|
|
|
ICCCM_GetGeoms(ewin);
|
|
|
|
return 1;
|
|
|
|
}
|
2007-04-16 09:36:10 -07:00
|
|
|
|
2023-10-21 02:04:12 -07:00
|
|
|
return 0;
|
2004-06-02 15:44:33 -07:00
|
|
|
}
|
2006-08-23 12:06:21 -07:00
|
|
|
|
|
|
|
#if USE_XSYNC
|
|
|
|
int
|
2023-10-21 02:04:12 -07:00
|
|
|
EwinSyncRequestSend(EWin *ewin)
|
2006-08-23 12:06:21 -07:00
|
|
|
{
|
2023-10-21 02:04:12 -07:00
|
|
|
long long count;
|
2006-08-23 12:06:21 -07:00
|
|
|
|
2023-10-21 02:04:12 -07:00
|
|
|
if (!ewin->ewmh.sync_request_enable || EServerIsGrabbed())
|
|
|
|
return 0;
|
2006-08-23 12:06:21 -07:00
|
|
|
|
2023-10-21 02:04:12 -07:00
|
|
|
count = ++ewin->ewmh.sync_request_count;
|
2006-08-23 12:06:21 -07:00
|
|
|
|
2023-10-21 02:04:12 -07:00
|
|
|
if (count == 0)
|
|
|
|
ewin->ewmh.sync_request_count = ++count;
|
|
|
|
ex_client_message32_send(EwinGetClientXwin(ewin),
|
|
|
|
ea_i.WM_PROTOCOLS,
|
|
|
|
StructureNotifyMask,
|
|
|
|
ea_n._NET_WM_SYNC_REQUEST,
|
|
|
|
Mode.events.time,
|
|
|
|
(long)(count & 0xffffffff), (long)(count >> 32),
|
|
|
|
0);
|
2006-08-23 12:06:21 -07:00
|
|
|
|
2023-10-21 02:04:12 -07:00
|
|
|
return 1;
|
2006-08-23 12:06:21 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2023-10-21 02:04:12 -07:00
|
|
|
EwinSyncRequestWait(EWin *ewin)
|
|
|
|
{
|
|
|
|
XSyncWaitCondition xswc[2];
|
|
|
|
unsigned int tus;
|
|
|
|
|
|
|
|
if (!ewin->ewmh.sync_request_enable || EServerIsGrabbed())
|
|
|
|
return;
|
|
|
|
|
|
|
|
xswc[0].trigger.counter = ewin->ewmh.sync_request_counter;
|
|
|
|
xswc[0].trigger.value_type = XSyncAbsolute;
|
|
|
|
XSyncIntsToValue(&xswc[0].trigger.wait_value,
|
|
|
|
ewin->ewmh.sync_request_count & 0xffffffff,
|
|
|
|
ewin->ewmh.sync_request_count >> 32);
|
|
|
|
xswc[0].trigger.test_type = XSyncPositiveComparison;
|
|
|
|
XSyncIntsToValue(&xswc[0].event_threshold, 0, 0);
|
|
|
|
|
|
|
|
xswc[1].trigger.counter = Mode.display.server_time;
|
|
|
|
xswc[1].trigger.value_type = XSyncRelative;
|
|
|
|
XSyncIntsToValue(&xswc[1].trigger.wait_value, 200, 0); /* 200 ms */
|
|
|
|
xswc[1].trigger.test_type = XSyncPositiveComparison;
|
|
|
|
XSyncIntsToValue(&xswc[1].event_threshold, 0, 0);
|
|
|
|
|
|
|
|
tus = GetTimeUs();
|
|
|
|
XSyncAwait(disp, xswc, 2);
|
|
|
|
if (EDebug(EDBUG_TYPE_SYNC))
|
|
|
|
Eprintf("%s: t=%#lx c=%llx: Delay=%u us\n", __func__,
|
|
|
|
xswc[0].trigger.counter, ewin->ewmh.sync_request_count,
|
|
|
|
GetTimeUs() - tus);
|
|
|
|
}
|
|
|
|
#endif /* USE_XSYNC */
|