e16/src/snaps.c

1694 lines
42 KiB
C

/*
* Copyright (C) 2000-2007 Carsten Haitzler, Geoff Harrison and various contributors
* Copyright (C) 2004-2020 Kim Woelders
*
* 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:
*
* 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.
*
* 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.
*/
#include "E.h"
#include "borders.h"
#include "desktops.h"
#include "dialog.h"
#include "ewins.h"
#include "file.h"
#include "groups.h"
#include "ipc.h"
#include "list.h"
#include "settings.h"
#include "snaps.h"
#include "timers.h"
#include "xwin.h"
struct _snapshot {
dlist_t list;
char *name;
char *win_title;
char *win_name;
char *win_class;
char *win_role;
EX_Window win;
EWin *used;
unsigned int startup_id;
char track_changes;
unsigned int match_flags;
unsigned int use_flags;
char *border_name;
int desktop;
int area_x, area_y;
int x, y;
int w, h;
int layer;
char sticky;
char shaded;
unsigned int flags[2];
char *cmd;
int *groups;
int num_groups;
char skiptask;
char skipfocus;
char skipwinlist;
#if USE_COMPOSITE
int opacity;
int focused_opacity;
char shadow;
#endif
};
static LIST_HEAD(ss_list);
static Timer *ss_timer = NULL;
static Snapshot *
_SnapCreate(const char *name)
{
Snapshot *sn;
sn = ECALLOC(Snapshot, 1);
if (!sn)
return NULL;
LIST_APPEND(Snapshot, &ss_list, sn);
sn->name = Estrdup(name);
return sn;
}
static void
_SnapDestroy(Snapshot * sn)
{
LIST_REMOVE(Snapshot, &ss_list, sn);
if (sn->used)
sn->used->snap = NULL;
Efree(sn->name);
Efree(sn->win_title);
Efree(sn->win_name);
Efree(sn->win_class);
Efree(sn->win_role);
Efree(sn->border_name);
Efree(sn->cmd);
Efree(sn->groups);
Efree(sn);
}
/*
* Stupid hack to fix apps that set WM_WINDOW_ROLE to
* a <name>-<pid>-<something>-<time> like thing.
* Is this even ICCCM compliant?
*/
static char *
_ParseRole(const char *role, char *buf, int len)
{
int l1, l2;
if (!role)
return NULL;
l1 = strlen(role);
if (l1 >= len)
l1 = len - 1;
for (l2 = l1; l2 > 0; l2--)
{
if (role[l2 - 1] != '-' &&
!(role[l2 - 1] >= '0' && role[l2 - 1] <= '9'))
break;
}
if (l1 - l2 > 8)
l1 = l2;
memcpy(buf, role, l1);
buf[l1] = '\0';
return buf;
}
#define SEQ(s1, s2) ((s1) && (s2) && !strcmp(s1, s2))
static int
_SnapEwinMatch(const Snapshot * sn, const EWin * ewin)
{
char buf[256], *s;
/* Don't allow matching anything */
if (!sn->match_flags)
return 0;
if (ewin->state.identified)
return sn->win == EwinGetClientXwin(ewin);
if (sn->startup_id && !sn->cmd)
return 0;
if (sn->match_flags & SNAP_MATCH_TITLE
&& !SEQ(sn->win_title, EwinGetIcccmName(ewin)))
return 0;
if (sn->match_flags & SNAP_MATCH_NAME
&& !SEQ(sn->win_name, EwinGetIcccmCName(ewin)))
return 0;
if (sn->match_flags & SNAP_MATCH_CLASS
&& !SEQ(sn->win_class, EwinGetIcccmClass(ewin)))
return 0;
if (sn->match_flags & SNAP_MATCH_ROLE)
{
s = _ParseRole(ewin->icccm.wm_role, buf, sizeof(buf));
if (!SEQ(sn->win_role, s))
return 0;
}
/* Match! */
return 1;
}
static int
_SnapEwinFindMatchCmd(const void *data, const void *match)
{
const Snapshot *sn = (Snapshot *) data;
const EWin *ewin = (EWin *) match;
return sn->used ||
!(sn->startup_id && SEQ(sn->cmd, ewin->icccm.wm_command) &&
_SnapEwinMatch(sn, ewin));
}
static int
_SnapEwinFindMatch(const void *data, const void *match)
{
const Snapshot *sn = (Snapshot *) data;
const EWin *ewin = (EWin *) match;
return sn->used || !_SnapEwinMatch(sn, ewin);
}
/* find a snapshot state that applies to this ewin */
static Snapshot *
_SnapEwinFind(EWin * ewin)
{
Snapshot *sn;
if (ewin->snap)
return ewin->snap;
if (LIST_IS_EMPTY(&ss_list))
return NULL;
/* If exec'ed by snap try matching command exactly */
sn = LIST_FIND(Snapshot, &ss_list, _SnapEwinFindMatchCmd, ewin);
if (sn && sn->startup_id > 0)
{
/* Assuming we were started by snap */
sn->startup_id = 0; /* Only the first time */
ewin->state.snapstarted = 1;
}
if (!sn)
sn = LIST_FIND(Snapshot, &ss_list, _SnapEwinFindMatch, ewin);
if (sn && !(sn->match_flags & SNAP_MATCH_MULTIPLE))
{
sn->used = ewin;
ewin->snap = sn;
}
return sn;
}
#define ST(s) ((s) ? (s) : "")
/* find a snapshot state that applies to this ewin Or if that doesnt exist */
/* create a new one */
static Snapshot *
_SnapEwinGet(EWin * ewin, unsigned int match_flags)
{
Snapshot *sn;
char buf[1024], *s;
sn = _SnapEwinFind(ewin);
if (sn)
return sn;
if ((match_flags & SNAP_MATCH_TITLE) && !EwinGetIcccmName(ewin))
match_flags ^= SNAP_MATCH_TITLE;
if ((match_flags & SNAP_MATCH_NAME) && !EwinGetIcccmCName(ewin))
match_flags ^= SNAP_MATCH_NAME;
if ((match_flags & SNAP_MATCH_CLASS) && !EwinGetIcccmClass(ewin))
match_flags ^= SNAP_MATCH_CLASS;
if ((match_flags & SNAP_MATCH_ROLE) && !ewin->icccm.wm_role)
match_flags ^= SNAP_MATCH_ROLE;
if (match_flags == 0)
{
if (!EwinGetIcccmName(ewin))
return NULL;
match_flags = SNAP_MATCH_TITLE;
}
sn = _SnapCreate(NULL);
if (!sn)
return NULL;
sn->match_flags = match_flags;
if (match_flags & SNAP_MATCH_TITLE)
sn->win_title = Estrdup(EwinGetIcccmName(ewin));
if (match_flags & SNAP_MATCH_NAME)
sn->win_name = Estrdup(EwinGetIcccmCName(ewin));
if (match_flags & SNAP_MATCH_CLASS)
sn->win_class = Estrdup(EwinGetIcccmClass(ewin));
if (match_flags & SNAP_MATCH_ROLE)
{
s = _ParseRole(ewin->icccm.wm_role, buf, sizeof(buf));
sn->win_role = Estrdup(s);
}
/* Set the snap name. Has no particular significance. */
if ((sn->win_name || sn->win_class) && sn->win_role)
Esnprintf(buf, sizeof(buf), "%s.%s:%s", ST(sn->win_name),
ST(sn->win_class), sn->win_role);
else if (sn->win_name || sn->win_class)
Esnprintf(buf, sizeof(buf), "%s.%s", ST(sn->win_name), ST(sn->win_class));
else if (sn->win_title)
Esnprintf(buf, sizeof(buf), "TITLE.%s", sn->win_title);
else /* We should not go here */
Esnprintf(buf, sizeof(buf), "TITLE.%s", EwinGetIcccmName(ewin));
sn->name = Estrdup(buf);
if (!(sn->match_flags & SNAP_MATCH_MULTIPLE))
{
sn->used = ewin;
ewin->snap = sn;
}
return sn;
}
/* record info about this Ewin's attributes */
static void
_SnapUpdateEwinBorder(Snapshot * sn, const EWin * ewin)
{
EFREE_DUP(sn->border_name, BorderGetName(ewin->normal_border));
}
static void
_SnapUpdateEwinDesktop(Snapshot * sn, const EWin * ewin)
{
sn->desktop = EoGetDeskNum(ewin);
}
static void
_SnapUpdateEwinSize(Snapshot * sn, const EWin * ewin)
{
sn->w = ewin->client.w;
sn->h = ewin->client.h;
}
static void
_SnapUpdateEwinLocation(Snapshot * sn, const EWin * ewin)
{
int ax, ay;
sn->x = EoGetX(ewin);
sn->y = EoGetY(ewin);
sn->area_x = ewin->area_x;
sn->area_y = ewin->area_y;
if (!EoIsSticky(ewin))
{
DeskGetArea(EoGetDesk(ewin), &ax, &ay);
sn->x += ((ax - sn->area_x) * WinGetW(VROOT));
sn->y += ((ay - sn->area_y) * WinGetH(VROOT));
}
}
static void
_SnapUpdateEwinLayer(Snapshot * sn, const EWin * ewin)
{
sn->layer = EoGetLayer(ewin);
}
static void
_SnapUpdateEwinSticky(Snapshot * sn, const EWin * ewin)
{
sn->sticky = EoIsSticky(ewin);
}
static void
_SnapUpdateEwinShade(Snapshot * sn, const EWin * ewin)
{
sn->shaded = ewin->state.shaded;
}
static void
_SnapUpdateEwinSkipLists(Snapshot * sn, const EWin * ewin)
{
sn->skiptask = ewin->props.skip_ext_task;
sn->skipwinlist = ewin->props.skip_winlist;
sn->skipfocus = ewin->props.skip_focuslist;
}
static void
_SnapUpdateEwinFlags(Snapshot * sn, const EWin * ewin)
{
EwinFlagsEncode(ewin, sn->flags);
}
static void
_SnapUpdateEwinCmd(Snapshot * sn, const EWin * ewin)
{
if (ewin->icccm.wm_machine &&
strcmp(ewin->icccm.wm_machine, Mode.wm.machine_name))
return;
EFREE_DUP(sn->cmd, ewin->icccm.wm_command);
}
static void
_SnapUpdateEwinGroups(Snapshot * sn, const EWin * ewin, char onoff)
{
EWin **gwins;
Group *const *groups;
int i, j, num, num_groups;
if (!ewin)
return;
if (!ewin->groups)
{
EFREE_NULL(sn->groups);
sn->num_groups = 0;
return;
}
gwins =
ListWinGroupMembersForEwin(ewin, GROUP_ACTION_ANY, Mode.nogroup, &num);
for (i = 0; i < num; i++)
{
if (onoff)
{
groups = EwinGetGroups(gwins[i], &num_groups);
if (!groups)
continue;
sn = gwins[i]->snap;
if (!sn)
sn = _SnapEwinGet(gwins[i], SNAP_MATCH_DEFAULT);
if (!sn)
continue;
sn->num_groups = num_groups;
EFREE_SET(sn->groups, EMALLOC(int, num_groups));
for (j = 0; j < num_groups; j++)
sn->groups[j] = GroupRemember(groups[j]);
}
else
{
if (ewin->snap)
{
sn = gwins[i]->snap;
if (sn)
{
EFREE_NULL(sn->groups);
sn->num_groups = 0;
}
}
}
}
Efree(gwins);
}
#if USE_COMPOSITE
static void
_SnapUpdateEwinOpacity(Snapshot * sn, const EWin * ewin)
{
sn->opacity = OpacityToPercent(ewin->props.opacity);
sn->focused_opacity = OpacityToPercent(ewin->props.focused_opacity);
}
static void
_SnapUpdateEwinShadow(Snapshot * sn, const EWin * ewin)
{
sn->shadow = EoGetShadow(ewin);
}
#endif
static void
_SnapUpdateEwin(Snapshot * sn, const EWin * ewin, unsigned int flags)
{
/* FIXME - We should check if anything is actually changed */
if (flags & SNAP_USE_BORDER)
_SnapUpdateEwinBorder(sn, ewin);
if (flags & SNAP_USE_COMMAND)
_SnapUpdateEwinCmd(sn, ewin);
if (flags & SNAP_USE_DESK)
_SnapUpdateEwinDesktop(sn, ewin);
if (flags & SNAP_USE_POS)
_SnapUpdateEwinLocation(sn, ewin);
if (flags & SNAP_USE_SIZE)
_SnapUpdateEwinSize(sn, ewin);
if (flags & SNAP_USE_LAYER)
_SnapUpdateEwinLayer(sn, ewin);
if (flags & SNAP_USE_STICKY)
_SnapUpdateEwinSticky(sn, ewin);
if (flags & SNAP_USE_SHADED)
_SnapUpdateEwinShade(sn, ewin);
if (flags & SNAP_USE_SKIP_LISTS)
_SnapUpdateEwinSkipLists(sn, ewin);
if (flags & SNAP_USE_FLAGS)
_SnapUpdateEwinFlags(sn, ewin);
#if USE_COMPOSITE
if (flags & SNAP_USE_OPACITY)
_SnapUpdateEwinOpacity(sn, ewin);
if (flags & SNAP_USE_SHADOW)
_SnapUpdateEwinShadow(sn, ewin);
#endif
if (flags & SNAP_USE_GROUPS)
_SnapUpdateEwinGroups(sn, ewin, ewin->num_groups);
SnapshotsSave();
}
static void
_EwinSnapSet(EWin * ewin, unsigned int match_flags, unsigned int use_flags)
{
Snapshot *sn;
/* Quit if nothing to be saved */
if (!match_flags || !(use_flags & SNAP_USE_ALL))
return;
sn = _SnapEwinGet(ewin, match_flags);
if (!sn)
return;
if (use_flags & SNAP_AUTO)
sn->track_changes = 1;
sn->use_flags = use_flags & SNAP_USE_ALL;
_SnapUpdateEwin(sn, ewin, use_flags);
}
void
SnapshotEwinUpdate(const EWin * ewin, unsigned int flags)
{
Snapshot *sn;
sn = ewin->snap;
if (!sn || !sn->track_changes)
return;
#if 0
Eprintf("%s: %s: %#x\n", __func__, EwinGetTitle(ewin), flags);
#endif
if (flags & sn->use_flags)
_SnapUpdateEwin(sn, ewin, flags);
}
/* unsnapshot any saved info about this ewin */
static void
_EwinSnapRemove(EWin * ewin)
{
if (ewin->snap)
_SnapDestroy(ewin->snap);
ewin->snap = NULL;
}
#if ENABLE_DIALOGS
/*
* Snapshot dialogs
*/
typedef struct {
EX_Window client;
struct {
char title;
char name;
char clss;
char role;
} match;
char track_changes;
char snap_border;
char snap_desktop;
char snap_size;
char snap_location;
char snap_layer;
char snap_sticky;
char snap_shaded;
char snap_cmd;
char snap_group;
char snap_skiplists;
char snap_flags;
#if USE_COMPOSITE
char snap_opacity;
char snap_shadow;
#endif
} SnapDlgData;
static void
_DlgApplySnap(Dialog * d, int val __UNUSED__, void *data __UNUSED__)
{
EWin *ewin;
SnapDlgData *sd = DLG_DATA_GET(d, SnapDlgData);
unsigned int match_flags, use_flags;
ewin = EwinFindByClient(sd->client);
if (!ewin)
goto done;
_EwinSnapRemove(ewin);
match_flags = 0;
if (sd->match.title)
match_flags |= SNAP_MATCH_TITLE;
if (sd->match.name)
match_flags |= SNAP_MATCH_NAME;
if (sd->match.clss)
match_flags |= SNAP_MATCH_CLASS;
if (sd->match.role)
match_flags |= SNAP_MATCH_ROLE;
if (!match_flags)
goto done;
use_flags = 0;
if (sd->track_changes)
use_flags |= SNAP_AUTO;
if (sd->snap_border)
use_flags |= SNAP_USE_BORDER;
if (sd->snap_cmd)
use_flags |= SNAP_USE_COMMAND;
if (sd->snap_desktop)
use_flags |= SNAP_USE_DESK;
if (sd->snap_location)
use_flags |= SNAP_USE_POS;
if (sd->snap_size)
use_flags |= SNAP_USE_SIZE;
if (sd->snap_layer)
use_flags |= SNAP_USE_LAYER;
if (sd->snap_sticky)
use_flags |= SNAP_USE_STICKY;
if (sd->snap_shaded)
use_flags |= SNAP_USE_SHADED;
if (sd->snap_skiplists)
use_flags |= SNAP_USE_SKIP_LISTS;
if (sd->snap_flags)
use_flags |= SNAP_USE_FLAGS;
#if USE_COMPOSITE
if (sd->snap_opacity)
use_flags |= SNAP_USE_OPACITY;
if (sd->snap_shadow)
use_flags |= SNAP_USE_SHADOW;
#endif
if (sd->snap_group)
use_flags |= SNAP_USE_GROUPS;
if (!use_flags)
goto done;
_EwinSnapSet(ewin, match_flags, use_flags);
done:
SnapshotsSave();
}
static void
_DlgFillSnap(Dialog * d, DItem * table, void *data)
{
SnapDlgData *sd = DLG_DATA_GET(d, SnapDlgData);
DItem *di;
Snapshot *sn;
char s[1024];
const EWin *ewin = (EWin *) data;
sd->client = EwinGetClientXwin(ewin);
sn = ewin->snap;
if (sn)
{
sd->match.title = (sn->match_flags & SNAP_MATCH_TITLE) != 0;
sd->match.name = (sn->match_flags & SNAP_MATCH_NAME) != 0;
sd->match.clss = (sn->match_flags & SNAP_MATCH_CLASS) != 0;
sd->match.role = (sn->match_flags & SNAP_MATCH_ROLE) != 0;
if (sn->track_changes)
sd->track_changes = 1;
if (sn->use_flags & SNAP_USE_BORDER)
sd->snap_border = 1;
if (sn->use_flags & SNAP_USE_COMMAND)
sd->snap_cmd = 1;
if (sn->use_flags & SNAP_USE_DESK)
sd->snap_desktop = 1;
if (sn->use_flags & SNAP_USE_POS)
sd->snap_location = 1;
if (sn->use_flags & SNAP_USE_SIZE)
sd->snap_size = 1;
if (sn->use_flags & SNAP_USE_LAYER)
sd->snap_layer = 1;
if (sn->use_flags & SNAP_USE_STICKY)
sd->snap_sticky = 1;
if (sn->use_flags & SNAP_USE_SHADED)
sd->snap_shaded = 1;
if (sn->use_flags & SNAP_USE_SKIP_LISTS)
sd->snap_skiplists = 1;
if (sn->use_flags & SNAP_USE_FLAGS)
sd->snap_flags = 1;
#if USE_COMPOSITE
if (sn->use_flags & SNAP_USE_OPACITY)
sd->snap_opacity = 1;
if (sn->use_flags & SNAP_USE_SHADOW)
sd->snap_shadow = 1;
#endif
if (sn->use_flags & SNAP_USE_GROUPS)
sd->snap_group = 1;
}
else
{
if (EwinGetIcccmCName(ewin))
{
sd->match.name = 1;
sd->match.clss = 1;
sd->match.role = !!ewin->icccm.wm_role;
}
else
{
sd->match.title = !!EwinGetIcccmName(ewin);
}
}
table = DialogAddItem(table, DITEM_TABLE);
DialogItemTableSetOptions(table, 4, 0, 0, 0);
di = DialogAddItem(table, DITEM_CHECKBUTTON);
DialogItemSetAlign(di, 0, 512);
DialogItemSetText(di, _("Title:"));
DialogItemCheckButtonSetPtr(di, &sd->match.title);
di = DialogAddItem(table, DITEM_TEXT);
DialogItemSetColSpan(di, 3);
DialogItemSetAlign(di, 1024, 512);
DialogItemSetText(di, EwinGetIcccmName(ewin));
if (EwinGetIcccmCName(ewin))
{
di = DialogAddItem(table, DITEM_CHECKBUTTON);
DialogItemSetAlign(di, 0, 512);
DialogItemSetText(di, _("Name:"));
DialogItemCheckButtonSetPtr(di, &sd->match.name);
di = DialogAddItem(table, DITEM_TEXT);
DialogItemSetColSpan(di, 3);
DialogItemSetAlign(di, 1024, 512);
DialogItemSetText(di, EwinGetIcccmCName(ewin));
}
if (EwinGetIcccmClass(ewin))
{
di = DialogAddItem(table, DITEM_CHECKBUTTON);
DialogItemSetAlign(di, 0, 512);
DialogItemSetText(di, _("Class:"));
DialogItemCheckButtonSetPtr(di, &sd->match.clss);
di = DialogAddItem(table, DITEM_TEXT);
DialogItemSetColSpan(di, 3);
DialogItemSetAlign(di, 1024, 512);
DialogItemSetText(di, EwinGetIcccmClass(ewin));
}
if (ewin->icccm.wm_role)
{
di = DialogAddItem(table, DITEM_CHECKBUTTON);
DialogItemSetAlign(di, 0, 512);
DialogItemSetText(di, _("Role:"));
DialogItemCheckButtonSetPtr(di, &sd->match.role);
di = DialogAddItem(table, DITEM_TEXT);
DialogItemSetColSpan(di, 3);
DialogItemSetAlign(di, 1024, 512);
DialogItemSetText(di, ewin->icccm.wm_role);
}
if (ewin->icccm.wm_command)
{
di = DialogAddItem(table, DITEM_TEXT);
DialogItemSetAlign(di, 0, 512);
DialogItemSetText(di, _("Command:"));
di = DialogAddItem(table, DITEM_TEXT);
DialogItemSetColSpan(di, 3);
DialogItemSetAlign(di, 1024, 512);
/* if the command is long, cut in into slices of about 80 characters */
if (strlen(ewin->icccm.wm_command) > 80)
{
int i = 0, slice, last;
s[0] = 0;
while ((i <= (int)strlen(ewin->icccm.wm_command))
&& (i < (int)(sizeof(s) / 4)))
{
last = i;
i += 64;
slice = 64;
/* and make sure that we don't cut in the middle of a word. */
while ((ewin->icccm.wm_command[i++] != ' ')
&& (i < (int)(sizeof(s) / 4)))
slice++;
strncat(s, ewin->icccm.wm_command + last, slice);
if (i < (int)(sizeof(s) / 4))
strcat(s, "\n");
else
strcat(s, "...\n");
}
DialogItemSetText(di, s);
}
else
DialogItemSetText(di, ewin->icccm.wm_command);
}
di = DialogAddItem(table, DITEM_SEPARATOR);
DialogItemSetColSpan(di, 4);
di = DialogAddItem(table, DITEM_CHECKBUTTON);
DialogItemSetColSpan(di, 4);
DialogItemSetText(di, _("Track Changes"));
DialogItemCheckButtonSetPtr(di, &sd->track_changes);
di = DialogAddItem(table, DITEM_CHECKBUTTON);
DialogItemSetColSpan(di, 2);
DialogItemSetText(di, _("Location"));
DialogItemCheckButtonSetPtr(di, &sd->snap_location);
di = DialogAddItem(table, DITEM_CHECKBUTTON);
DialogItemSetColSpan(di, 2);
DialogItemSetText(di, _("Border style"));
DialogItemCheckButtonSetPtr(di, &sd->snap_border);
di = DialogAddItem(table, DITEM_CHECKBUTTON);
DialogItemSetColSpan(di, 2);
DialogItemSetText(di, _("Size"));
DialogItemCheckButtonSetPtr(di, &sd->snap_size);
di = DialogAddItem(table, DITEM_CHECKBUTTON);
DialogItemSetColSpan(di, 2);
DialogItemSetText(di, _("Desktop"));
DialogItemCheckButtonSetPtr(di, &sd->snap_desktop);
di = DialogAddItem(table, DITEM_CHECKBUTTON);
DialogItemSetColSpan(di, 2);
DialogItemSetText(di, _("Shaded state"));
DialogItemCheckButtonSetPtr(di, &sd->snap_shaded);
di = DialogAddItem(table, DITEM_CHECKBUTTON);
DialogItemSetColSpan(di, 2);
DialogItemSetText(di, _("Sticky state"));
DialogItemCheckButtonSetPtr(di, &sd->snap_sticky);
di = DialogAddItem(table, DITEM_CHECKBUTTON);
DialogItemSetColSpan(di, 2);
DialogItemSetText(di, _("Stacking layer"));
DialogItemCheckButtonSetPtr(di, &sd->snap_layer);
di = DialogAddItem(table, DITEM_CHECKBUTTON);
DialogItemSetColSpan(di, 2);
DialogItemSetText(di, _("Window List Skip"));
DialogItemCheckButtonSetPtr(di, &sd->snap_skiplists);
#if USE_COMPOSITE
di = DialogAddItem(table, DITEM_CHECKBUTTON);
DialogItemSetColSpan(di, 2);
DialogItemSetText(di, _("Opacity"));
DialogItemCheckButtonSetPtr(di, &sd->snap_opacity);
di = DialogAddItem(table, DITEM_CHECKBUTTON);
DialogItemSetColSpan(di, 2);
DialogItemSetText(di, _("Shadowing"));
DialogItemCheckButtonSetPtr(di, &sd->snap_shadow);
#endif
di = DialogAddItem(table, DITEM_CHECKBUTTON);
DialogItemSetColSpan(di, 2);
DialogItemSetText(di, _("Flags"));
DialogItemCheckButtonSetPtr(di, &sd->snap_flags);
di = DialogAddItem(table, DITEM_NONE);
DialogItemSetColSpan(di, 2);
if (ewin->icccm.wm_command)
{
char ok = 1;
if (ewin->icccm.wm_machine)
{
if (strcmp(ewin->icccm.wm_machine, Mode.wm.machine_name))
ok = 0;
}
if (ok)
{
di = DialogAddItem(table, DITEM_CHECKBUTTON);
DialogItemSetColSpan(di, 4);
DialogItemSetText(di, _("Restart application on login"));
DialogItemCheckButtonSetPtr(di, &sd->snap_cmd);
}
else
{
di = DialogAddItem(table, DITEM_NONE);
DialogItemSetColSpan(di, 4);
}
}
else
{
di = DialogAddItem(table, DITEM_NONE);
DialogItemSetColSpan(di, 4);
}
if (ewin->groups)
{
di = DialogAddItem(table, DITEM_CHECKBUTTON);
DialogItemSetColSpan(di, 4);
DialogItemSetText(di, _("Remember this window's group(s)"));
DialogItemCheckButtonSetPtr(di, &sd->snap_group);
}
}
static const DialogDef DlgSnap = {
NULL,
NULL, N_("Remembered Application Attributes"),
sizeof(SnapDlgData),
SOUND_NONE,
"pix/snapshots.png",
N_("Select the attributes of this\n"
"window you wish to Remember\n" "from now on\n"),
_DlgFillSnap,
DLG_OAC, _DlgApplySnap, NULL
};
static void
_EwinSnapDialog(const EWin * ewin)
{
char s[1024];
Esnprintf(s, sizeof(s), "SNAPSHOT_WINDOW-%#x", EwinGetClientXwin(ewin));
DialogShowSimpleWithName(&DlgSnap, s, (void *)ewin);
}
/* list of remembered items for the remember dialog -- it's either
* _another_ global var, or a wrapper struct to pass data to the
* callback funcs besides the dialog itself -- this is much easier */
typedef struct _remwinlist {
Snapshot *snap;
char remove;
} RememberWinList;
static RememberWinList *rd_ewin_list;
static void
_DlgApplyRemember(Dialog * d __UNUSED__,
int val __UNUSED__, void *data __UNUSED__)
{
int i;
if (!rd_ewin_list)
return;
for (i = 0; rd_ewin_list[i].snap; i++)
{
if (!rd_ewin_list[i].remove)
continue;
_SnapDestroy(rd_ewin_list[i].snap);
}
/* save snapshot info to disk */
SnapshotsSave();
}
static void
_DlgExitRemember(Dialog * d __UNUSED__)
{
EFREE_NULL(rd_ewin_list);
}
static void
CB_RememberWindowSettings(Dialog * d __UNUSED__, int val __UNUSED__, void *data)
{
RememberWinList *rd = (RememberWinList *) data;
Snapshot *sn;
if (!rd)
return;
/* Make sure its still there */
sn = LIST_CHECK(Snapshot, &ss_list, rd->snap);
if (!sn || !sn->used)
return;
_EwinSnapDialog(sn->used);
}
static void
_DlgFillRemember(Dialog * d __UNUSED__, DItem * table, void *data __UNUSED__)
{
DItem *di;
Snapshot *sn;
int i, num;
char buf[128];
const char *s;
DialogItemTableSetOptions(table, 3, 0, 0, 0);
num = LIST_GET_COUNT(&ss_list);
rd_ewin_list = EMALLOC(RememberWinList, num + 1);
if (num > 0)
{
di = DialogAddItem(table, DITEM_TEXT);
DialogItemSetColSpan(di, 3);
DialogItemSetFill(di, 0, 0);
DialogItemSetAlign(di, 0, 512);
DialogItemSetText(di, _("Delete"));
}
i = 0;
LIST_FOR_EACH(Snapshot, &ss_list, sn)
{
rd_ewin_list[i].snap = sn;
rd_ewin_list[i].remove = 0;
di = DialogAddItem(table, DITEM_CHECKBUTTON);
DialogItemSetColSpan(di, 2);
DialogItemSetAlign(di, 0, 512);
if (sn->used)
s = EwinGetTitle(sn->used);
else if (sn->win_title)
s = sn->win_title;
else
{
Esnprintf(buf, sizeof(buf), "%s.%s", sn->win_name, sn->win_class);
s = buf;
}
DialogItemSetText(di, s);
DialogItemCheckButtonSetPtr(di, &(rd_ewin_list[i].remove));
if (sn->used)
{
di = DialogAddItem(table, DITEM_BUTTON);
DialogItemSetAlign(di, 0, 512);
DialogItemSetText(di, _("Remembered Settings..."));
DialogItemSetCallback(di, CB_RememberWindowSettings, 0,
(char *)(&rd_ewin_list[i]));
}
else
{
di = DialogAddItem(table, DITEM_TEXT);
DialogItemSetText(di, _("Unused"));
}
i++;
}
rd_ewin_list[num].snap = NULL;
/* finish remember window */
if (!num)
{
di = DialogAddItem(table, DITEM_TEXT);
DialogItemSetColSpan(di, 3);
DialogItemSetText(di,
_
("There are no active windows with remembered attributes."));
}
}
const DialogDef DlgRemember = {
"CONFIGURE_PAGER",
N_("Remember"), N_("Remembered Windows Settings"),
0,
SOUND_SETTINGS_PAGER,
"pix/snapshots.png",
N_("Enlightenment Remembered\n" "Windows Settings Dialog"),
_DlgFillRemember,
DLG_OC, _DlgApplyRemember, _DlgExitRemember
};
#endif /* ENABLE_DIALOGS */
/* ... combine writes, only save after a timeout */
static int
_SnapshotsSaveReal(void *data __UNUSED__)
{
SnapshotsSaveReal();
return 0;
}
void
SnapshotsSave(void)
{
TIMER_DEL(ss_timer);
TIMER_ADD(ss_timer, 5000, _SnapshotsSaveReal, NULL);
}
/* save out all snapped info to disk */
void
SnapshotsSaveReal(void)
{
Snapshot *sn;
int j;
char buf[4096], s[4096];
FILE *f;
if (!Mode.wm.save_ok)
goto done;
Etmp(s);
f = fopen(s, "w");
if (!f)
goto done;
LIST_FOR_EACH(Snapshot, &ss_list, sn)
{
fprintf(f, "NEW: %s\n", sn->name);
if (sn->used)
fprintf(f, "WIN: %#x\n", EwinGetClientXwin(sn->used));
if ((sn->match_flags & SNAP_MATCH_TITLE) && sn->win_title)
fprintf(f, "TITLE: %s\n", sn->win_title);
if ((sn->match_flags & SNAP_MATCH_NAME) && sn->win_name)
fprintf(f, "NAME: %s\n", sn->win_name);
if ((sn->match_flags & SNAP_MATCH_CLASS) && sn->win_class)
fprintf(f, "CLASS: %s\n", sn->win_class);
if ((sn->match_flags & SNAP_MATCH_ROLE) && sn->win_role)
fprintf(f, "ROLE: %s\n", sn->win_role);
if (sn->track_changes)
fprintf(f, "AUTO: yes\n");
if ((sn->use_flags & SNAP_USE_BORDER) && sn->border_name)
fprintf(f, "BORDER: %s\n", sn->border_name);
if ((sn->use_flags & SNAP_USE_COMMAND) && sn->cmd)
fprintf(f, "CMD: %s\n", sn->cmd);
if (sn->use_flags & SNAP_USE_DESK)
fprintf(f, "DESKTOP: %i\n", sn->desktop);
if (sn->use_flags & SNAP_USE_POS)
fprintf(f, "RES: %i %i\n", WinGetW(VROOT), WinGetH(VROOT));
if (sn->use_flags & SNAP_USE_SIZE)
fprintf(f, "WH: %i %i\n", sn->w, sn->h);
if (sn->use_flags & SNAP_USE_POS)
fprintf(f, "XY: %i %i %i %i\n", sn->x, sn->y, sn->area_x, sn->area_y);
if (sn->use_flags & SNAP_USE_LAYER)
fprintf(f, "LAYER: %i\n", sn->layer);
if (sn->use_flags & SNAP_USE_STICKY)
fprintf(f, "STICKY: %i\n", sn->sticky);
if (sn->use_flags & SNAP_USE_SHADED)
fprintf(f, "SHADE: %i\n", sn->shaded);
if (sn->use_flags & SNAP_USE_SKIP_LISTS)
{
fprintf(f, "SKIPTASK: %i\n", sn->skiptask);
fprintf(f, "SKIPWINLIST: %i\n", sn->skipwinlist);
fprintf(f, "SKIPFOCUS: %i\n", sn->skipfocus);
}
if (sn->use_flags & SNAP_USE_FLAGS)
fprintf(f, "FLAGS: %#x %#x\n", sn->flags[0], sn->flags[1]);
#if USE_COMPOSITE
if (sn->use_flags & SNAP_USE_OPACITY)
fprintf(f, "OPACITY: %i %i\n", sn->opacity, sn->focused_opacity);
if (sn->use_flags & SNAP_USE_SHADOW)
fprintf(f, "SHADOW: %i\n", sn->shadow);
#endif
if (sn->groups)
{
for (j = 0; j < sn->num_groups; j++)
fprintf(f, "GROUP: %i\n", sn->groups[j]);
}
fprintf(f, "\n");
}
fclose(f);
Esnprintf(buf, sizeof(buf), "%s.snapshots", EGetSavePrefix());
if (EDebug(EDBUG_TYPE_SNAPS))
Eprintf("%s: %s\n", __func__, buf);
E_mv(s, buf);
if (!isfile(buf))
Alert(_("Error saving snaps file"));
GroupsSave();
done:
TIMER_DEL(ss_timer);
}
void
SnapshotsSpawn(void)
{
Snapshot *sn;
LIST_FOR_EACH(Snapshot, &ss_list, sn)
{
if ((sn->use_flags & SNAP_USE_COMMAND) && (sn->cmd) &&
!sn->used && !(sn->match_flags & SNAP_MATCH_MULTIPLE))
{
sn->startup_id = ++Mode.apps.startup_id;
Espawn(sn->cmd);
}
}
}
/* load all snapped info */
static int
_SnapshotsLoad(FILE * fs)
{
Snapshot *sn = NULL;
char buf[4096], *s;
int res_w, res_h, a, b, c, d;
res_w = WinGetW(VROOT);
res_h = WinGetH(VROOT);
while (fgets(buf, sizeof(buf), fs))
{
s = strchr(buf, ':');
if (!s)
continue;
*s++ = '\0';
s = Estrtrim(s);
if (!buf[0] || !s[0])
continue;
if (!strcmp(buf, "NEW"))
{
res_w = WinGetW(VROOT);
res_h = WinGetH(VROOT);
sn = _SnapCreate(s);
}
else if (sn)
{
if (!strcmp(buf, "WIN"))
{
sn->win = strtoul(s, NULL, 0);
}
else if (!strcmp(buf, "TITLE"))
{
sn->win_title = Estrdup(s);
sn->match_flags |= SNAP_MATCH_TITLE;
}
else if (!strcmp(buf, "NAME"))
{
sn->win_name = Estrdup(s);
sn->match_flags |= SNAP_MATCH_NAME;
}
else if (!strcmp(buf, "CLASS"))
{
sn->win_class = Estrdup(s);
sn->match_flags |= SNAP_MATCH_CLASS;
}
else if (!strcmp(buf, "ROLE"))
{
sn->win_role = Estrdup(s);
sn->match_flags |= SNAP_MATCH_ROLE;
}
else if (!strcmp(buf, "AUTO"))
{
sn->track_changes = 1;
}
else if (!strcmp(buf, "BORDER"))
{
sn->use_flags |= SNAP_USE_BORDER;
sn->border_name = Estrdup(s);
}
else if (!strcmp(buf, "CMD"))
{
sn->use_flags |= SNAP_USE_COMMAND;
sn->cmd = Estrdup(s);
}
else if (!strcmp(buf, "DESKTOP"))
{
sn->use_flags |= SNAP_USE_DESK;
sn->desktop = atoi(s);
}
else if (!strcmp(buf, "RES"))
{
if (sscanf(s, "%u %u", &a, &b) < 2)
continue;
if (a <= 0 || b <= 0)
continue;
res_w = a;
res_h = b;
}
else if (!strcmp(buf, "WH"))
{
if (sscanf(s, "%u %u", &a, &b) < 2)
continue;
if (a <= 0 || b <= 0)
continue;
sn->use_flags |= SNAP_USE_SIZE;
sn->w = a;
sn->h = b;
}
else if (!strcmp(buf, "XY"))
{
if (sscanf(s, "%d %d %u %u", &a, &b, &c, &d) < 4)
continue;
sn->use_flags |= SNAP_USE_POS;
sn->x = a;
sn->y = b;
/* we changed reses since we last used this snapshot file */
if (res_w != WinGetW(VROOT))
{
if (sn->use_flags & SNAP_USE_SIZE)
{
if ((res_w - sn->w) <= 0)
sn->x = 0;
else
sn->x =
(sn->x * (WinGetW(VROOT) - sn->w)) /
(res_w - sn->w);
}
else
{
if (sn->x >= WinGetW(VROOT))
sn->x = WinGetW(VROOT) - 32;
}
}
if (res_h != WinGetH(VROOT))
{
if (sn->use_flags & SNAP_USE_SIZE)
{
if ((res_h - sn->h) <= 0)
sn->y = 0;
else
sn->y =
(sn->y * (WinGetH(VROOT) - sn->h)) /
(res_h - sn->h);
}
else
{
if (sn->y >= WinGetH(VROOT))
sn->y = WinGetH(VROOT) - 32;
}
}
sn->area_x = c;
sn->area_y = d;
}
else if (!strcmp(buf, "LAYER"))
{
sn->use_flags |= SNAP_USE_LAYER;
sn->layer = atoi(s);
}
else if (!strcmp(buf, "STICKY"))
{
sn->use_flags |= SNAP_USE_STICKY;
sn->sticky = atoi(s);
}
else if (!strcmp(buf, "SHADE"))
{
sn->use_flags |= SNAP_USE_SHADED;
sn->shaded = atoi(s);
}
else if (!strcmp(buf, "SKIPFOCUS"))
{
sn->use_flags |= SNAP_USE_SKIP_LISTS;
sn->skipfocus = atoi(s);
}
else if (!strcmp(buf, "SKIPTASK"))
{
sn->use_flags |= SNAP_USE_SKIP_LISTS;
sn->skiptask = atoi(s);
}
else if (!strcmp(buf, "SKIPWINLIST"))
{
sn->use_flags |= SNAP_USE_SKIP_LISTS;
sn->skipwinlist = atoi(s);
}
else if (!strcmp(buf, "FLAGS"))
{
sn->use_flags |= SNAP_USE_FLAGS;
sn->flags[0] = sn->flags[1] = 0;
sscanf(s, "%i %i", sn->flags, sn->flags + 1);
}
else if (!strcmp(buf, "GROUP"))
{
sn->use_flags |= SNAP_USE_GROUPS;
sn->num_groups++;
sn->groups = EREALLOC(int, sn->groups, sn->num_groups);
sn->groups[sn->num_groups - 1] = atoi(s);
GroupRememberByGid(sn->groups[sn->num_groups - 1]);
}
#if USE_COMPOSITE
else if (!strcmp(buf, "OPACITY"))
{
sn->use_flags |= SNAP_USE_OPACITY;
a = 100;
b = 100;
sscanf(s, "%i %i", &a, &b);
if (b == 1)
b = 100; /* BW compat - focused is opaque */
sn->opacity = a;
sn->focused_opacity = b;
}
else if (!strcmp(buf, "SHADOW"))
{
sn->use_flags |= SNAP_USE_SHADOW;
sn->shadow = atoi(s);
}
#endif
}
}
return 0;
}
void
SnapshotsLoad(void)
{
char s[4096];
GroupsLoad();
Esnprintf(s, sizeof(s), "%s.snapshots", EGetSavePrefix());
ConfigFileLoad(s, NULL, _SnapshotsLoad, 0);
}
/* make a client window conform to snapshot info */
void
SnapshotEwinApply(EWin * ewin)
{
Snapshot *sn;
int ax, ay;
unsigned int use_flags;
_SnapEwinFind(ewin); /* Find a saved settings match */
sn = ewin->snap;
if (!sn)
{
if (ewin->props.autosave)
_EwinSnapSet(ewin, SNAP_MATCH_DEFAULT, SNAP_USE_ALL | SNAP_AUTO);
return;
}
if (ewin->props.autosave)
sn->track_changes = 1;
use_flags = sn->use_flags;
/* If restarting don't override stuff set in attributes/properties */
if (ewin->state.identified)
use_flags &= SNAP_USE_LAYER | SNAP_USE_SHADOW | SNAP_USE_GROUPS |
SNAP_USE_OPACITY;
if (use_flags & SNAP_USE_STICKY)
EoSetSticky(ewin, sn->sticky);
if (use_flags & SNAP_USE_DESK)
EoSetDesk(ewin, DeskGetValid(sn->desktop));
if (use_flags & SNAP_USE_SIZE)
{
ewin->client.w = sn->w;
ewin->client.h = sn->h;
ewin->state.maximized_horz = ewin->state.maximized_vert = 0;
}
if (use_flags & SNAP_USE_POS)
{
ewin->state.placed = 1;
ewin->client.x = sn->x;
ewin->client.y = sn->y;
ewin->icccm.grav = NorthWestGravity;
#if 0 /* No, do later in EwinDetermineArea() */
ewin->area_x = sn->area_x;
ewin->area_y = sn->area_y;
#endif
if (!EoIsSticky(ewin))
{
DeskGetArea(EoGetDesk(ewin), &ax, &ay);
ewin->client.x += ((sn->area_x - ax) * WinGetW(VROOT));
ewin->client.y += ((sn->area_y - ay) * WinGetH(VROOT));
}
}
if (use_flags & SNAP_USE_LAYER)
EoSetLayer(ewin, sn->layer);
if (use_flags & SNAP_USE_SKIP_LISTS)
{
ewin->props.skip_focuslist = sn->skipfocus;
ewin->props.skip_ext_task = sn->skiptask;
ewin->props.skip_winlist = sn->skipwinlist;
}
if (use_flags & SNAP_USE_FLAGS)
EwinFlagsDecode(ewin, sn->flags);
if (use_flags & SNAP_USE_SHADED)
ewin->state.shaded = sn->shaded;
if (use_flags & SNAP_USE_BORDER)
EwinBorderSetInitially(ewin, sn->border_name);
if (use_flags & SNAP_USE_GROUPS)
GroupsEwinAdd(ewin, sn->groups, sn->num_groups);
#if USE_COMPOSITE
if (use_flags & SNAP_USE_OPACITY)
{
sn->opacity = OpacityFix(sn->opacity, 0);
sn->focused_opacity = OpacityFix(sn->focused_opacity, 0);
ewin->props.opacity = OpacityFromPercent(sn->opacity);
ewin->props.focused_opacity = OpacityFromPercent(sn->focused_opacity);
ewin->ewmh.opacity_update = 1; /* Set opacity on client window */
}
if (use_flags & SNAP_USE_SHADOW)
EoSetShadow(ewin, sn->shadow);
#endif
if (EDebug(EDBUG_TYPE_SNAPS))
Eprintf("Snap get snap %#x: %4d+%4d %4dx%4d: %s\n",
EwinGetClientXwin(ewin), ewin->client.x, ewin->client.y,
ewin->client.w, ewin->client.h, EwinGetTitle(ewin));
}
/* Detach snapshot from ewin */
void
SnapshotEwinUnmatch(EWin * ewin)
{
Snapshot *sn;
sn = ewin->snap;
if (!sn)
return;
ewin->snap = NULL;
sn->used = NULL;
}
void
SnapshotEwinParse(EWin * ewin, const char *params)
{
char param[1024];
const char *p;
int len;
unsigned int match_flags, use_flags;
p = params;
if (!p)
return;
match_flags = SNAP_MATCH_DEFAULT;
use_flags = 0;
for (;;)
{
param[0] = '\0';
len = 0;
sscanf(p, "%s %n", param, &len);
if (len <= 0)
break;
p += len;
if (!strcmp(param, "all"))
{
use_flags = SNAP_USE_ALL;
}
#if ENABLE_DIALOGS
else if (!strcmp(param, "dialog"))
{
_EwinSnapDialog(ewin);
break;
}
#endif
else if (!strcmp(param, "none"))
_EwinSnapRemove(ewin);
else if (!strcmp(param, "auto"))
use_flags |= SNAP_AUTO;
else if (!strcmp(param, "border"))
use_flags |= SNAP_USE_BORDER;
else if (!strcmp(param, "command"))
use_flags |= SNAP_USE_COMMAND;
else if (!strcmp(param, "desktop"))
use_flags |= SNAP_USE_DESK;
else if (!strcmp(param, "location"))
use_flags |= SNAP_USE_POS;
else if (!strcmp(param, "size"))
use_flags |= SNAP_USE_SIZE;
else if (!strcmp(param, "layer"))
use_flags |= SNAP_USE_LAYER;
else if (!strcmp(param, "shade"))
use_flags |= SNAP_USE_SHADED;
else if (!strcmp(param, "sticky"))
use_flags |= SNAP_USE_STICKY;
#if USE_COMPOSITE
else if (!strcmp(param, "opacity"))
use_flags |= SNAP_USE_OPACITY;
else if (!strcmp(param, "shadow"))
use_flags |= SNAP_USE_SHADOW;
#endif
else if (!strcmp(param, "group"))
use_flags |= SNAP_USE_GROUPS;
}
if (ewin->snap)
{
match_flags = ewin->snap->match_flags;
use_flags |= ewin->snap->use_flags;
}
_EwinSnapSet(ewin, match_flags, use_flags);
SnapshotsSave();
}
/*
* IPC functions
* A bit ugly...
*/
const char SnapshotsIpcText[] =
"usage:\n" " list_remember [full]\n"
" Retrieve a list of remembered windows. with full, the list\n"
" includes the window's remembered attributes\n";
#define SS(s) ((s) ? (s) : NoText)
static const char NoText[] = "-NONE-";
static void
_SnapShow(void *data, void *prm)
{
Snapshot *sn = (Snapshot *) data;
int full = !!prm;
char buf[FILEPATH_LEN_MAX];
const char *name;
name = (sn->name) ? sn->name : "???";
if (!full)
{
if (sn->used)
IpcPrintf("%s\n", name);
else
IpcPrintf("%s (unused)\n", name);
return;
}
#define SU(sn, item) ((sn->match_flags & item) ? '>' : ':')
if (sn->used)
Esnprintf(buf, sizeof(buf), "In use - %#x", EwinGetClientXwin(sn->used));
else
Esnprintf(buf, sizeof(buf), "*** Unused ***");
IpcPrintf(" Snapshot Name: %s %s\n", name, buf);
if (sn->win_title)
IpcPrintf(" Window Title%c %s\n", SU(sn, SNAP_MATCH_TITLE),
sn->win_title);
if (sn->win_name)
IpcPrintf(" Window Name%c %s\n", SU(sn, SNAP_MATCH_NAME),
sn->win_name);
if (sn->win_class)
IpcPrintf(" Window Class%c %s\n", SU(sn, SNAP_MATCH_CLASS),
sn->win_class);
if (sn->win_role)
IpcPrintf(" Window Role%c %s\n", SU(sn, SNAP_MATCH_ROLE),
sn->win_role);
if (sn->track_changes)
IpcPrintf(" Tracking changes\n");
if (sn->use_flags & SNAP_USE_BORDER)
IpcPrintf(" Border Name: %s\n", SS(sn->border_name));
if (sn->use_flags & SNAP_USE_DESK)
IpcPrintf(" desktop: %d\n", sn->desktop);
if (sn->use_flags & SNAP_USE_POS)
IpcPrintf(" (x, y): %d, %d area (x, y): %d, %d\n",
sn->x, sn->y, sn->area_x, sn->area_y);
if (sn->use_flags & SNAP_USE_SIZE)
IpcPrintf(" (w, h): %d, %d\n", sn->w, sn->h);
if (sn->use_flags & SNAP_USE_LAYER)
IpcPrintf(" layer: %d\n", sn->layer);
if (sn->use_flags & SNAP_USE_STICKY)
IpcPrintf(" sticky: %d\n", sn->sticky);
if (sn->use_flags & SNAP_USE_SHADED)
IpcPrintf(" shade: %d\n", sn->shaded);
if (sn->use_flags & SNAP_USE_COMMAND)
IpcPrintf(" command: %s\n", SS(sn->cmd));
if (sn->use_flags & SNAP_USE_SKIP_LISTS)
IpcPrintf
(" skiptask: %d skipfocus: %d skipwinlist: %d\n",
sn->skiptask, sn->skipfocus, sn->skipwinlist);
if (sn->use_flags & SNAP_USE_FLAGS)
IpcPrintf(" flags: %#x %#x\n", sn->flags[0], sn->flags[1]);
IpcPrintf("\n");
}
void
SnapshotsIpcFunc(const char *params)
{
const char *p;
char cmd[128], prm[4096];
int len;
Snapshot *sn;
cmd[0] = prm[0] = '\0';
p = params;
if (p)
{
len = 0;
sscanf(p, "%100s %4000s %n", cmd, prm, &len);
p += len;
}
if (LIST_IS_EMPTY(&ss_list))
{
IpcPrintf("No remembered windows\n");
return;
}
if (!p || cmd[0] == '?')
{
LIST_FOR_EACH(Snapshot, &ss_list, sn) _SnapShow(sn, NULL);
}
else
{
LIST_FOR_EACH(Snapshot, &ss_list, sn) _SnapShow(sn, (void *)1L);
}
}