Newer
Older
#ifdef HAVE_RANDR
XRRUpdateConfiguration (ev);
#else
xscreen->width = ev->width;
xscreen->height = ev->height;
#endif
placeSidewalks (params.wrap_workspaces);
clientScreenResize ();
static inline void
handleConfigureRequest (XConfigureRequestEvent * ev)
Olivier Fourdan
committed
Client *c = NULL;
TRACE ("entering handleConfigureRequest");
TRACE ("ConfigureRequest on window (0x%lx)", ev->window);
while (XCheckTypedWindowEvent (dpy, ev->window, ConfigureRequest,
&otherEvent))
if (otherEvent.xconfigurerequest.value_mask == ev->value_mask)
{
ev = &otherEvent.xconfigurerequest;
}
else
{
XPutBackEvent (dpy, &otherEvent);
break;
}
wc.x = ev->x;
wc.y = ev->y;
wc.width = ev->width;
wc.height = ev->height;
wc.sibling = ev->above;
wc.stack_mode = ev->detail;
wc.border_width = ev->border_width;
c = clientGetFromWindow (ev->window, WINDOW);
if (!c)
{
1048
1049
1050
1051
1052
1053
1054
1055
1056
1057
1058
1059
1060
1061
1062
1063
1064
1065
1066
1067
1068
1069
1070
1071
1072
1073
1074
/* Some app tend or try to manipulate the wm frame to achieve fullscreen mode */
c = clientGetFromWindow (ev->window, FRAME);
if (c)
{
TRACE ("client %s (0x%lx) is attempting to manipulate its frame!",
c->name, c->window);
if (ev->value_mask & CWX)
{
wc.x += frameLeft (c);
}
if (ev->value_mask & CWY)
{
wc.y += frameTop (c);
}
if (ev->value_mask & CWWidth)
{
wc.width -= frameLeft (c) + frameRight (c);
}
if (ev->value_mask & CWHeight)
{
wc.height -= frameTop (c) + frameBottom (c);
}
/* We don't allow changing stacking order by accessing the frame
window because that would break the layer management in xfwm4
*/
ev->value_mask &= ~(CWSibling | CWStackMode);
}
}
if (c)
{
gboolean constrained = FALSE;
TRACE ("handleConfigureRequest managed window \"%s\" (0x%lx)",
c->name, c->window);
if (FLAG_TEST (c->flags, CLIENT_FLAG_MOVING_RESIZING))
{
/* Sorry, but it's not the right time for configure request */
return;
}
if (c->type == WINDOW_DESKTOP)
{
/* Ignore stacking request for DESKTOP windows */
ev->value_mask &= ~(CWSibling | CWStackMode);
}
clientCoordGravitate (c, APPLY, &wc.x, &wc.y);
if (FLAG_TEST (c->flags, CLIENT_FLAG_FULLSCREEN))
{
int cx, cy;
/* size request from fullscreen windows get fullscreen */
cx = frameX (c) + (frameWidth (c) / 2);
cy = frameY (c) + (frameHeight (c) / 2);
wc.x = MyDisplayX (cx, cy);
wc.y = MyDisplayY (cx, cy);
wc.width = MyDisplayWidth (dpy, screen, cx, cy);
wc.height = MyDisplayHeight (dpy, screen, cx, cy);
ev->value_mask |= (CWX | CWY | CWWidth | CWHeight);
}
/* Clean up buggy requests that set all flags */
if ((ev->value_mask & CWX) && (wc.x == c->x))
{
ev->value_mask &= ~CWX;
}
if ((ev->value_mask & CWY) && (wc.y == c->y))
{
ev->value_mask &= ~CWY;
}
if ((ev->value_mask & CWWidth) && (wc.width == c->width))
{
ev->value_mask &= ~CWWidth;
}
if ((ev->value_mask & CWHeight) && (wc.height == c->height))
{
ev->value_mask &= ~CWHeight;
}
/* Still a move/resize after cleanup? */
if (ev->value_mask & (CWX | CWY | CWWidth | CWHeight))
{
if (FLAG_TEST (c->flags, CLIENT_FLAG_MAXIMIZED))
{
clientRemoveMaximizeFlag (c);
}
constrained = TRUE;
}
if (ev->value_mask & CWStackMode)
{
clientPassGrabButton1 (NULL);
}
Olivier Fourdan
committed
#if 0
/* Let's say that if the client performs a XRaiseWindow, we show the window if hidden */
if ((ev->value_mask & CWStackMode) && (wc.stack_mode == Above))
{
if ((c->win_workspace == workspace) ||
(FLAG_TEST (c->flags, CLIENT_FLAG_STICKY)))
if (FLAG_TEST (c->flags, CLIENT_FLAG_HIDDEN))
{
clientShow (c, TRUE);
}
}
}
Olivier Fourdan
committed
#endif
clientConfigure (c, &wc, ev->value_mask,
(constrained ? CFG_CONSTRAINED : 0) | CFG_REQUEST);
TRACE ("unmanaged configure request for win 0x%lx", ev->window);
XConfigureWindow (dpy, ev->window, ev->value_mask, &wc);
static inline void
handleEnterNotify (XCrossingEvent * ev)
Olivier Fourdan
committed
Client *c = NULL;
TRACE ("entering handleEnterNotify");
if ((ev->mode == NotifyGrab) || (ev->mode == NotifyUngrab)
|| (ev->detail > NotifyNonlinearVirtual))
/* We're not interested in such notifications */
return;
TRACE ("EnterNotify on window (0x%lx)", ev->window);
c = clientGetFromWindow (ev->window, FRAME);
if (c && !(params.click_to_focus) && (clientAcceptFocus (c)))
TRACE ("EnterNotify window is \"%s\"", c->name);
if ((c->type != WINDOW_DOCK) && (c->type != WINDOW_DESKTOP))
{
Olivier Fourdan
committed
clientSetFocus (c, FOCUS_SORT);
Olivier Fourdan
committed
if (!(params.raise_on_click))
{
Olivier Fourdan
committed
}
static inline void
handleLeaveNotify (XCrossingEvent * ev)
Olivier Fourdan
committed
{
TRACE ("entering handleLeaveNotify");
Olivier Fourdan
committed
Olivier Fourdan
committed
if ((ev->mode == NotifyGrab) || (ev->mode == NotifyUngrab)
|| (ev->detail > NotifyNonlinearVirtual))
{
/* We're not interested in such notifications */
return;
}
if ((ev->window == sidewalk[0]) || (ev->window == sidewalk[1]))
Olivier Fourdan
committed
{
TRACE ("Reset edge_scroll_x");
edge_scroll_x = 0;
}
Olivier Fourdan
committed
}
static inline void
handleFocusIn (XFocusChangeEvent * ev)
Olivier Fourdan
committed
Client *c = NULL;
TRACE ("entering handleFocusIn");
Olivier Fourdan
committed
TRACE ("handleFocusIn (0x%lx) mode = %s",
ev->window,
(ev->mode == NotifyNormal) ?
"NotifyNormal" :
(ev->mode == NotifyWhileGrabbed) ?
"NotifyWhileGrabbed" :
"(unknown)");
Olivier Fourdan
committed
TRACE ("handleFocusIn (0x%lx) detail = %s",
ev->window,
(ev->detail == NotifyAncestor) ?
"NotifyAncestor" :
(ev->detail == NotifyVirtual) ?
"NotifyVirtual" :
(ev->detail == NotifyInferior) ?
"NotifyInferior" :
(ev->detail == NotifyNonlinear) ?
"NotifyNonlinear" :
(ev->detail == NotifyNonlinearVirtual) ?
"NotifyNonlinearVirtual" :
(ev->detail == NotifyPointer) ?
"NotifyPointer" :
(ev->detail == NotifyPointerRoot) ?
"NotifyPointerRoot" :
(ev->detail == NotifyDetailNone) ?
"NotifyDetailNone" :
"(unknown)");
if ((ev->mode == NotifyGrab) || (ev->mode == NotifyUngrab)
|| (ev->detail > NotifyNonlinearVirtual))
{
/* We're not interested in such notifications */
return;
}
c = clientGetFromWindow (ev->window, WINDOW);
TRACE ("FocusIn on window (0x%lx)", ev->window);
TRACE ("focus set to \"%s\" (0x%lx)", c->name, c->window);
Olivier Fourdan
committed
clientUpdateFocus (c, FOCUS_NONE);
if (params.raise_on_focus && !params.click_to_focus)
{
reset_timeout ();
}
static inline void
handleFocusOut (XFocusChangeEvent * ev)
Olivier Fourdan
committed
Client *c = NULL;
Client *c2 = NULL;
TRACE ("entering handleFocusOut");
1272
1273
1274
1275
1276
1277
1278
1279
1280
1281
1282
1283
1284
1285
1286
1287
1288
1289
1290
1291
1292
1293
1294
1295
1296
1297
TRACE ("handleFocusOut (0x%lx) mode = %s",
ev->window,
(ev->mode == NotifyNormal) ?
"NotifyNormal" :
(ev->mode == NotifyWhileGrabbed) ?
"NotifyWhileGrabbed" :
"(unknown)");
TRACE ("handleFocusOut (0x%lx) detail = %s",
ev->window,
(ev->detail == NotifyAncestor) ?
"NotifyAncestor" :
(ev->detail == NotifyVirtual) ?
"NotifyVirtual" :
(ev->detail == NotifyInferior) ?
"NotifyInferior" :
(ev->detail == NotifyNonlinear) ?
"NotifyNonlinear" :
(ev->detail == NotifyNonlinearVirtual) ?
"NotifyNonlinearVirtual" :
(ev->detail == NotifyPointer) ?
"NotifyPointer" :
(ev->detail == NotifyPointerRoot) ?
"NotifyPointerRoot" :
(ev->detail == NotifyDetailNone) ?
"NotifyDetailNone" :
"(unknown)");
Olivier Fourdan
committed
if ((ev->mode == NotifyNormal)
&& ((ev->detail == NotifyNonlinear)
|| (ev->detail == NotifyNonlinearVirtual)))
Olivier Fourdan
committed
{
c = clientGetFromWindow (ev->window, WINDOW);
Olivier Fourdan
committed
c2 = clientGetFocus ();
TRACE ("FocusOut on window (0x%lx)", ev->window);
Olivier Fourdan
committed
if ((c) && (c == c2))
{
TRACE ("focus lost from \"%s\" (0x%lx)", c->name, c->window);
Olivier Fourdan
committed
clientUpdateFocus (NULL, FOCUS_NONE);
clientPassGrabButton1 (NULL);
/* Clear timeout */
clear_timeout ();
}
Olivier Fourdan
committed
else if ((c2) && (ev->window == root) &&
(ev->detail == NotifyNonlinearVirtual))
{
/* Handle focus transition for Motif combo menu */
clientSetFocus (c2, FOCUS_FORCE);
}
Olivier Fourdan
committed
}
static inline void
handlePropertyNotify (XPropertyEvent * ev)
Olivier Fourdan
committed
Client *c = NULL;
TRACE ("entering handlePropertyNotify");
c = clientGetFromWindow (ev->window, WINDOW);
if (c)
{
if (ev->atom == XA_WM_NORMAL_HINTS)
{
TRACE
("client \"%s\" (0x%lx) has received a XA_WM_NORMAL_HINTS notify",
c->name, c->window);
}
else if ((ev->atom == XA_WM_NAME) || (ev->atom == net_wm_name))
{
TRACE ("client \"%s\" (0x%lx) has received a XA_WM_NAME notify",
c->name, c->window);
if (c->name)
{
free (c->name);
}
getWindowName (dpy, c->window, &c->name);
FLAG_SET (c->flags, CLIENT_FLAG_NAME_CHANGED);
frameDraw (c, TRUE, FALSE);
}
else if (ev->atom == motif_wm_hints)
{
TRACE
("client \"%s\" (0x%lx) has received a motif_wm_hints notify",
c->name, c->window);
}
else if (ev->atom == XA_WM_HINTS)
{
TRACE ("client \"%s\" (0x%lx) has received a XA_WM_HINTS notify",
c->name, c->window);
c->wmhints = XGetWMHints (dpy, c->window);
if (c->wmhints)
{
if (c->wmhints->flags & WindowGroupHint)
{
c->group_leader = c->wmhints->window_group;
}
else if (ev->atom == wm_protocols)
{
TRACE
("client \"%s\" (0x%lx) has received a wm_protocols notify",
c->name, c->window);
clientGetWMProtocols (c);
}
else if (ev->atom == win_hints)
{
TRACE ("client \"%s\" (0x%lx) has received a win_hints notify",
c->name, c->window);
getHint (dpy, c->window, win_hints, &c->win_hints);
1384
1385
1386
1387
1388
1389
1390
1391
1392
1393
1394
1395
1396
1397
1398
1399
1400
1401
1402
1403
1404
1405
1406
}
else if (ev->atom == net_wm_window_type)
{
TRACE
("client \"%s\" (0x%lx) has received a net_wm_window_type notify",
c->name, c->window);
clientGetNetWmType (c);
frameDraw (c, TRUE, FALSE);
}
else if (ev->atom == net_wm_strut)
{
TRACE ("client \"%s\" (0x%lx) has received a net_wm_strut notify",
c->name, c->window);
clientGetNetStruts (c);
}
else if (ev->atom == wm_colormap_windows)
{
clientUpdateColormaps (c);
if (c == clientGetFocus ())
{
clientInstallColormaps (c);
}
}
#ifdef HAVE_STARTUP_NOTIFICATION
else if (ev->atom == net_startup_id)
{
if (c->startup_id)
{
free (c->startup_id);
c->startup_id = NULL;
}
getWindowStartupId (dpy, c->window, &c->startup_id);
}
#endif
else if (ev->atom == gnome_panel_desktop_area)
TRACE ("root has received a gnome_panel_desktop_area notify");
getGnomeDesktopMargins (dpy, screen, gnome_margins);
workspaceUpdateArea (margins, gnome_margins);
static inline void
handleClientMessage (XClientMessageEvent * ev)
Olivier Fourdan
committed
Client *c = NULL;
TRACE ("entering handleClientMessage");
/* Don't get surprised with the multiple "if (!clientIsTransientOrModal(c))" tests
xfwm4 really treats transient differently
*/
c = clientGetFromWindow (ev->window, WINDOW);
if (c)
{
if ((ev->message_type == wm_change_state) && (ev->format == 32)
&& (ev->data.l[0] == IconicState))
{
TRACE
("client \"%s\" (0x%lx) has received a wm_change_state event",
c->name, c->window);
if (!FLAG_TEST (c->flags, CLIENT_FLAG_HIDDEN) &&
CLIENT_CAN_HIDE_WINDOW (c))
{
clientHide (c, c->win_workspace, TRUE);
}
}
else if ((ev->message_type == win_state) && (ev->format == 32))
{
TRACE ("client \"%s\" (0x%lx) has received a win_state event",
c->name, c->window);
clientUpdateWinState (c, ev);
}
else if ((ev->message_type == win_layer) && (ev->format == 32))
{
TRACE ("client \"%s\" (0x%lx) has received a win_layer event",
c->name, c->window);
if ((ev->data.l[0] != c->win_layer) && !clientIsTransientOrModal (c))
{
clientSetLayer (c, ev->data.l[0]);
clientSetNetState (c);
}
}
else if ((ev->message_type == win_workspace) && (ev->format == 32))
{
TRACE ("client \"%s\" (0x%lx) has received a win_workspace event",
c->name, c->window);
if ((ev->data.l[0] != c->win_workspace) && !clientIsTransientOrModal (c))
{
clientSetWorkspace (c, ev->data.l[0], TRUE);
}
}
else if ((ev->message_type == net_wm_desktop) && (ev->format == 32))
{
TRACE
("client \"%s\" (0x%lx) has received a net_wm_desktop event",
c->name, c->window);
{
if (ev->data.l[0] == ALL_WORKSPACES)
{
if (FLAG_TEST_AND_NOT (c->flags, CLIENT_FLAG_HAS_STICK,
CLIENT_FLAG_STICKY))
{
clientStick (c, TRUE);
frameDraw (c, FALSE, FALSE);
}
}
else
{
if (FLAG_TEST_ALL (c->flags,
1498
1499
1500
1501
1502
1503
1504
1505
1506
1507
1508
1509
1510
1511
1512
1513
1514
1515
1516
1517
1518
1519
1520
1521
1522
1523
1524
1525
1526
1527
1528
1529
1530
1531
1532
1533
1534
1535
1536
1537
1538
1539
1540
1541
CLIENT_FLAG_HAS_STICK | CLIENT_FLAG_STICKY))
{
clientUnstick (c, TRUE);
frameDraw (c, FALSE, FALSE);
}
if (ev->data.l[0] != c->win_workspace)
{
clientSetWorkspace (c, ev->data.l[0], TRUE);
}
}
}
}
else if ((ev->message_type == net_close_window) && (ev->format == 32))
{
TRACE
("client \"%s\" (0x%lx) has received a net_close_window event",
c->name, c->window);
clientClose (c);
}
else if ((ev->message_type == net_wm_state) && (ev->format == 32))
{
TRACE ("client \"%s\" (0x%lx) has received a net_wm_state event",
c->name, c->window);
clientUpdateNetState (c, ev);
}
else if ((ev->message_type == net_wm_moveresize)
&& (ev->format == 32))
{
TRACE
("client \"%s\" (0x%lx) has received a net_wm_moveresize event",
c->name, c->window);
g_message (_("%s: Operation not supported (yet)\n"),
g_get_prgname ());
/* TBD */
}
else if ((ev->message_type == net_active_window)
&& (ev->format == 32))
{
TRACE
("client \"%s\" (0x%lx) has received a net_active_window event",
c->name, c->window);
workspaceSwitch (c->win_workspace, NULL);
clientShow (c, TRUE);
clientRaise (c);
Olivier Fourdan
committed
clientSetFocus (c, FOCUS_SORT);
1548
1549
1550
1551
1552
1553
1554
1555
1556
1557
1558
1559
1560
1561
1562
1563
1564
1565
1566
1567
1568
if (((ev->message_type == win_workspace)
|| (ev->message_type == net_current_desktop))
&& (ev->format == 32))
{
TRACE
("root has received a win_workspace or a net_current_desktop event");
if (ev->data.l[0] != workspace)
{
workspaceSwitch (ev->data.l[0], NULL);
}
}
else if (((ev->message_type == win_workspace_count)
|| (ev->message_type == net_number_of_desktops))
&& (ev->format == 32))
{
TRACE ("root has received a win_workspace_count event");
if (ev->data.l[0] != params.workspace_count)
{
workspaceSetCount (ev->data.l[0]);
}
}
else if ((ev->message_type == net_system_tray_manager)
&& (ev->data.l[1] == net_system_tray_selection)
&& (ev->format == 32))
{
TRACE ("root has received a net_system_tray_manager event");
systray = getSystrayWindow (dpy);
}
else
{
TRACE ("unidentified client message for window 0x%lx",
ev->window);
}
static inline void
handleShape (XShapeEvent * ev)
Olivier Fourdan
committed
Client *c = NULL;
TRACE ("entering handleShape");
c = clientGetFromWindow (ev->window, WINDOW);
if (c)
frameDraw (c, FALSE, TRUE);
static inline void
handleColormapNotify (XColormapEvent * ev)
Olivier Fourdan
committed
Client *c = NULL;
Olivier Fourdan
committed
TRACE ("entering handleColormapNotify");
Olivier Fourdan
committed
c = clientGetFromWindow (ev->window, WINDOW);
if ((c) && (ev->window == c->window) && (ev->new))
if (c == clientGetFocus ())
{
clientInstallColormaps (c);
}
void
handleEvent (XEvent * ev)
TRACE ("entering handleEvent");
sn_process_event (ev);
1623
1624
1625
1626
1627
1628
1629
1630
1631
1632
1633
1634
1635
1636
1637
1638
1639
1640
1641
1642
1643
case MotionNotify:
handleMotionNotify ((XMotionEvent *) ev);
break;
case KeyPress:
handleKeyPress ((XKeyEvent *) ev);
break;
case ButtonPress:
handleButtonPress ((XButtonEvent *) ev);
break;
case ButtonRelease:
handleButtonRelease ((XButtonEvent *) ev);
break;
case DestroyNotify:
handleDestroyNotify ((XDestroyWindowEvent *) ev);
break;
case UnmapNotify:
handleUnmapNotify ((XUnmapEvent *) ev);
break;
case MapRequest:
handleMapRequest ((XMapRequestEvent *) ev);
break;
case MapNotify:
handleMapNotify ((XMapEvent *) ev);
break;
case ConfigureNotify:
handleConfigureNotify ((XConfigureEvent *) ev);
break;
1650
1651
1652
1653
1654
1655
1656
1657
1658
1659
1660
1661
1662
1663
1664
1665
1666
1667
1668
1669
1670
1671
1672
1673
1674
1675
1676
1677
1678
case ConfigureRequest:
handleConfigureRequest ((XConfigureRequestEvent *) ev);
break;
case EnterNotify:
handleEnterNotify ((XCrossingEvent *) ev);
break;
case LeaveNotify:
handleLeaveNotify ((XCrossingEvent *) ev);
break;
case FocusIn:
handleFocusIn ((XFocusChangeEvent *) ev);
break;
case FocusOut:
handleFocusOut ((XFocusChangeEvent *) ev);
break;
case PropertyNotify:
handlePropertyNotify ((XPropertyEvent *) ev);
break;
case ClientMessage:
handleClientMessage ((XClientMessageEvent *) ev);
break;
case ColormapNotify:
handleColormapNotify ((XColormapEvent *) ev);
break;
default:
if (shape && (ev->type == shape_event))
{
handleShape ((XShapeEvent *) ev);
}
}
if (!gdk_events_pending () && !XPending (dpy))
{
if (reload)
{
reloadSettings (UPDATE_ALL);
reload = FALSE;
}
else if (quit)
{
gtk_main_quit ();
}
GtkToXEventFilterStatus
xfwm4_event_filter (XEvent * xevent, gpointer data)
TRACE ("entering xfwm4_event_filter");
handleEvent (xevent);
TRACE ("leaving xfwm4_event_filter");
/* GTK specific stuff */
static void
menu_callback (Menu * menu, MenuOp op, Window client_xwindow,
gpointer menu_data, gpointer item_data)
Olivier Fourdan
committed
TRACE ("entering menu_callback");
if (menu_event_window)
removeTmpEventWin (menu_event_window);
menu_event_window = None;
if (menu_data)
c = (Client *) menu_data;
c = clientGetFromWindow (c->window, WINDOW);
if (!c)
{
menu_free (menu);
return;
}
c->button_pressed[MENU_BUTTON] = FALSE;
Olivier Fourdan
committed
1733
1734
1735
1736
1737
1738
1739
1740
1741
1742
1743
1744
1745
1746
1747
1748
1749
1750
1751
1752
1753
1754
1755
1756
1757
1758
1759
1760
1761
1762
1763
1764
1765
case MENU_OP_QUIT:
gtk_main_quit ();
break;
case MENU_OP_MAXIMIZE:
case MENU_OP_UNMAXIMIZE:
if (CLIENT_CAN_MAXIMIZE_WINDOW (c))
{
clientToggleMaximized (c, WIN_STATE_MAXIMIZED);
}
break;
case MENU_OP_MINIMIZE:
if (CLIENT_CAN_HIDE_WINDOW (c))
{
clientHide (c, c->win_workspace, TRUE);
}
frameDraw (c, FALSE, FALSE);
break;
case MENU_OP_MINIMIZE_ALL:
clientHideAll (c, c->win_workspace);
frameDraw (c, FALSE, FALSE);
break;
case MENU_OP_UNMINIMIZE:
clientShow (c, TRUE);
break;
case MENU_OP_SHADE:
case MENU_OP_UNSHADE:
clientToggleShaded (c);
break;
case MENU_OP_STICK:
case MENU_OP_UNSTICK:
clientToggleSticky (c, TRUE);
frameDraw (c, FALSE, FALSE);
break;
case MENU_OP_WORKSPACES:
clientSetWorkspace (c, GPOINTER_TO_INT (item_data), TRUE);
frameDraw (c, FALSE, FALSE);
break;
case MENU_OP_DELETE:
frameDraw (c, FALSE, FALSE);
clientClose (c);
break;
Olivier Fourdan
committed
case MENU_OP_CONTEXT_HELP:
clientEnterContextMenuState (c);
frameDraw (c, FALSE, FALSE);
break;
case MENU_OP_ABOVE:
case MENU_OP_NORMAL:
clientToggleAbove (c);
/* Fall thru */
default:
frameDraw (c, FALSE, FALSE);
break;
}
menu_free (menu);
static gboolean
show_popup_cb (GtkWidget * widget, GdkEventButton * ev, gpointer data)
Olivier Fourdan
committed
Menu *menu;
MenuOp ops;
MenuOp insensitive;
Client *c = NULL;
gint x = ev->x_root;
gint y = ev->y_root;
TRACE ("entering show_popup_cb");
if (((ev->button == 1) || (ev->button == 3)) && (c = (Client *) data))
{
c->button_pressed[MENU_BUTTON] = TRUE;
frameDraw (c, FALSE, FALSE);
y = c->y;
ops = MENU_OP_DELETE | MENU_OP_MINIMIZE_ALL | MENU_OP_WORKSPACES;
insensitive = 0;
if (!FLAG_TEST (c->flags, CLIENT_FLAG_HAS_CLOSE))
{
insensitive |= MENU_OP_DELETE;
}
if (FLAG_TEST (c->flags, CLIENT_FLAG_MAXIMIZED))
{
ops |= MENU_OP_UNMAXIMIZE;
if (!CLIENT_CAN_MAXIMIZE_WINDOW (c))
{
insensitive |= MENU_OP_UNMAXIMIZE;
}
}
else
{
ops |= MENU_OP_MAXIMIZE;
if (!CLIENT_CAN_MAXIMIZE_WINDOW (c))
{
insensitive |= MENU_OP_MAXIMIZE;
}
}
if (FLAG_TEST (c->flags, CLIENT_FLAG_HIDDEN))
{
ops |= MENU_OP_UNMINIMIZE;
if (!CLIENT_CAN_HIDE_WINDOW (c))
{
insensitive |= MENU_OP_UNMINIMIZE;
}
}
else
{
ops |= MENU_OP_MINIMIZE;
if (!CLIENT_CAN_HIDE_WINDOW (c))
{
insensitive |= MENU_OP_MINIMIZE;
}
}
if (FLAG_TEST (c->flags, CLIENT_FLAG_SHADED))
{
ops |= MENU_OP_UNSHADE;
}
else
{
ops |= MENU_OP_SHADE;
}
if (FLAG_TEST (c->flags, CLIENT_FLAG_STICKY))
{
ops |= MENU_OP_UNSTICK;
if (!FLAG_TEST (c->flags, CLIENT_FLAG_HAS_STICK))
{
insensitive |= MENU_OP_UNSTICK;
}
}
else
{
ops |= MENU_OP_STICK;
if (!FLAG_TEST (c->flags, CLIENT_FLAG_HAS_STICK))
{
insensitive |= MENU_OP_STICK;
}
}
Olivier Fourdan
committed
/* KDE extension */
clientGetWMProtocols(c);
if (FLAG_TEST (c->wm_flags, WM_FLAG_CONTEXT_HELP))
{
ops |= MENU_OP_CONTEXT_HELP;
}
if (FLAG_TEST(c->flags, CLIENT_FLAG_ABOVE))
{
ops |= MENU_OP_NORMAL;
if (clientIsTransientOrModal (c) ||
FLAG_TEST (c->flags, CLIENT_FLAG_BELOW | CLIENT_FLAG_FULLSCREEN))
{
insensitive |= MENU_OP_NORMAL;
}
}
else
{
ops |= MENU_OP_ABOVE;
if (clientIsTransientOrModal (c) ||
FLAG_TEST (c->flags, CLIENT_FLAG_BELOW | CLIENT_FLAG_FULLSCREEN))
{
insensitive |= MENU_OP_ABOVE;
}
}
|| !FLAG_TEST (c->flags, CLIENT_FLAG_HAS_STICK)
|| FLAG_TEST (c->flags, CLIENT_FLAG_STICKY))
{
insensitive |= MENU_OP_WORKSPACES;
}
Olivier Fourdan
committed
}
else
{
Olivier Fourdan
committed
}
if (button_handler_id)
Olivier Fourdan
committed
{
g_signal_handler_disconnect (GTK_OBJECT (getDefaultGtkWidget ()),
button_handler_id);
Olivier Fourdan
committed
}
button_handler_id =
g_signal_connect (GTK_OBJECT (getDefaultGtkWidget ()),
"button_press_event", GTK_SIGNAL_FUNC (show_popup_cb),
(gpointer) NULL);
Olivier Fourdan
committed
if (menu_event_window)
removeTmpEventWin (menu_event_window);
menu_event_window = None;
/*
Since all button press/release events are catched by the windows frames, there is some
side effect with GTK menu. When a menu is opened, any click on the window frame is not
detected as a click outside the menu, and the menu doesn't close.
To avoid this (painless but annoying) behaviour, we just setup a no event window that
"hides" the events to regular windows.
That might look tricky, but it's very efficient and save plenty of lines of complicated
Don't forget to delete that window once the menu is closed, though, or we'll get in
Olivier Fourdan
committed
menu_event_window = setTmpEventWin (0, 0,
MyDisplayFullWidth (dpy, screen),
MyDisplayFullHeight (dpy, screen),
NoEventMask);
menu =
menu_default (ops, insensitive, menu_callback, c->win_workspace,
params.workspace_count, c);
if (!menu_popup (menu, x, y, ev->button, ev->time))
{
TRACE ("Cannot open menu");
gdk_beep ();
c->button_pressed[MENU_BUTTON] = FALSE;
frameDraw (c, FALSE, FALSE);
removeTmpEventWin (menu_event_window);
menu_event_window = None;
menu_free (menu);
Olivier Fourdan
committed
}
Olivier Fourdan
committed
return (TRUE);
static gboolean
set_reload (void)
("setting reload flag so all prefs will be reread at next event loop");
reload = TRUE;
static gboolean
dbl_click_time (void)
Olivier Fourdan
committed
TRACE ("setting dbl_click_time");
Olivier Fourdan
committed
g_value_init (&tmp_val, G_TYPE_INT);
if (gdk_setting_get ("gtk-double-click-time", &tmp_val))
params.dbl_click_time = abs (g_value_get_int (&tmp_val));
}
return (TRUE);
}
static gboolean
client_event_cb (GtkWidget * widget, GdkEventClient * ev)
TRACE ("entering client_event_cb");
Olivier Fourdan
committed
if (!atom_rcfiles)
Olivier Fourdan
committed
{
atom_rcfiles = gdk_atom_intern ("_GTK_READ_RCFILES", FALSE);
Olivier Fourdan
committed
}
if (ev->message_type == atom_rcfiles)
Olivier Fourdan
committed
{
Olivier Fourdan
committed
}
return (FALSE);