2002-10-02 16:26:05 +00:00
|
|
|
// Ewmh.cc for fluxbox
|
2004-01-21 09:37:10 +00:00
|
|
|
// Copyright (c) 2002-2004 Henrik Kinnunen (fluxgen at user.sourceforge.net)
|
2002-10-02 16:26:05 +00: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:
|
|
|
|
//
|
|
|
|
// The above copyright notice and this permission notice shall be included in
|
|
|
|
// all copies or substantial portions of the Software.
|
|
|
|
//
|
|
|
|
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
|
|
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
2003-03-04 11:06:03 +00:00
|
|
|
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
|
2002-10-02 16:26:05 +00:00
|
|
|
// THE AUTHORS OR COPYRIGHT HOLDERS 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.
|
|
|
|
|
2004-06-20 04:49:33 +00:00
|
|
|
// $Id: Ewmh.cc,v 1.47 2004/06/20 04:49:32 rathnor Exp $
|
2002-10-02 16:26:05 +00:00
|
|
|
|
|
|
|
#include "Ewmh.hh"
|
|
|
|
|
|
|
|
#include "Screen.hh"
|
|
|
|
#include "Window.hh"
|
2003-04-14 12:13:36 +00:00
|
|
|
#include "WinClient.hh"
|
2003-04-25 11:15:01 +00:00
|
|
|
#include "Workspace.hh"
|
2003-12-19 00:36:53 +00:00
|
|
|
#include "fluxbox.hh"
|
2002-10-02 16:26:05 +00:00
|
|
|
|
2004-01-19 18:21:51 +00:00
|
|
|
#include "FbTk/FbWindow.hh"
|
2004-06-07 11:46:05 +00:00
|
|
|
#include "FbTk/I18n.hh"
|
2004-01-19 18:21:51 +00:00
|
|
|
|
2002-10-02 16:26:05 +00:00
|
|
|
#include <iostream>
|
2003-04-14 12:13:36 +00:00
|
|
|
#include <algorithm>
|
2002-11-26 19:49:40 +00:00
|
|
|
#include <new>
|
2002-10-02 16:26:05 +00:00
|
|
|
using namespace std;
|
|
|
|
|
|
|
|
Ewmh::Ewmh() {
|
2002-12-01 13:42:15 +00:00
|
|
|
createAtoms();
|
2002-10-02 16:26:05 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
Ewmh::~Ewmh() {
|
2002-12-01 13:42:15 +00:00
|
|
|
while (!m_windows.empty()) {
|
2003-03-04 11:06:03 +00:00
|
|
|
XDestroyWindow(FbTk::App::instance()->display(), m_windows.back());
|
2002-12-01 13:42:15 +00:00
|
|
|
m_windows.pop_back();
|
|
|
|
}
|
2002-10-02 16:26:05 +00:00
|
|
|
}
|
|
|
|
|
2003-03-03 21:51:13 +00:00
|
|
|
void Ewmh::initForScreen(BScreen &screen) {
|
2003-03-04 21:41:05 +00:00
|
|
|
Display *disp = FbTk::App::instance()->display();
|
2003-03-04 11:06:03 +00:00
|
|
|
|
2002-10-02 16:26:05 +00:00
|
|
|
|
2002-12-01 13:42:15 +00:00
|
|
|
Window wincheck = XCreateSimpleWindow(disp,
|
2003-05-10 22:59:32 +00:00
|
|
|
screen.rootWindow().window(),
|
2003-04-15 12:22:06 +00:00
|
|
|
0, 0, 5, 5, 0, 0, 0);
|
2002-10-16 23:32:17 +00:00
|
|
|
|
2002-12-01 13:42:15 +00:00
|
|
|
if (wincheck != None) {
|
2003-08-11 14:51:15 +00:00
|
|
|
// store the window so we can delete it later
|
2002-12-01 13:42:15 +00:00
|
|
|
m_windows.push_back(wincheck);
|
2002-10-02 16:26:05 +00:00
|
|
|
|
2003-05-19 22:43:48 +00:00
|
|
|
screen.rootWindow().changeProperty(m_net_supporting_wm_check, XA_WINDOW, 32,
|
|
|
|
PropModeReplace, (unsigned char *) &wincheck, 1);
|
2002-12-01 13:42:15 +00:00
|
|
|
XChangeProperty(disp, wincheck, m_net_supporting_wm_check, XA_WINDOW, 32,
|
2002-10-16 23:32:17 +00:00
|
|
|
PropModeReplace, (unsigned char *) &wincheck, 1);
|
2002-10-02 16:26:05 +00:00
|
|
|
|
2002-12-01 13:42:15 +00:00
|
|
|
XChangeProperty(disp, wincheck, m_net_wm_name, XA_STRING, 8,
|
2002-10-02 16:26:05 +00:00
|
|
|
PropModeReplace, (unsigned char *) "Fluxbox", strlen("Fluxbox"));
|
2002-12-01 13:42:15 +00:00
|
|
|
}
|
2002-10-02 16:26:05 +00:00
|
|
|
|
2002-12-01 13:42:15 +00:00
|
|
|
//set supported atoms
|
|
|
|
Atom atomsupported[] = {
|
|
|
|
// window properties
|
2003-06-18 13:34:30 +00:00
|
|
|
m_net_wm_strut,
|
2002-12-01 13:42:15 +00:00
|
|
|
m_net_wm_state,
|
|
|
|
// states that we support:
|
|
|
|
m_net_wm_state_sticky,
|
|
|
|
m_net_wm_state_shaded,
|
2003-08-27 21:06:04 +00:00
|
|
|
m_net_wm_state_maximized_horz,
|
|
|
|
m_net_wm_state_maximized_vert,
|
2003-09-23 13:52:05 +00:00
|
|
|
m_net_wm_state_fullscreen,
|
2004-01-18 19:14:08 +00:00
|
|
|
m_net_wm_state_hidden,
|
|
|
|
m_net_wm_state_skip_taskbar,
|
2003-09-23 13:52:05 +00:00
|
|
|
|
2004-01-21 09:37:10 +00:00
|
|
|
// window type
|
|
|
|
m_net_wm_window_type,
|
|
|
|
m_net_wm_window_type_dock,
|
2004-06-20 04:49:33 +00:00
|
|
|
m_net_wm_window_type_desktop,
|
2004-01-21 09:37:10 +00:00
|
|
|
|
2002-12-01 13:42:15 +00:00
|
|
|
// root properties
|
|
|
|
m_net_client_list,
|
2004-01-19 18:21:51 +00:00
|
|
|
m_net_client_list_stacking,
|
2002-12-01 13:42:15 +00:00
|
|
|
m_net_number_of_desktops,
|
|
|
|
m_net_current_desktop,
|
|
|
|
m_net_active_window,
|
|
|
|
m_net_close_window,
|
2004-02-14 12:15:35 +00:00
|
|
|
m_net_moveresize_window,
|
2004-01-19 18:21:51 +00:00
|
|
|
m_net_workarea,
|
|
|
|
|
|
|
|
// desktop properties
|
|
|
|
m_net_wm_desktop,
|
2002-12-01 13:42:15 +00:00
|
|
|
m_net_desktop_names,
|
2004-01-19 18:21:51 +00:00
|
|
|
m_net_desktop_viewport,
|
|
|
|
m_net_desktop_geometry,
|
|
|
|
|
2002-12-01 13:42:15 +00:00
|
|
|
m_net_supporting_wm_check
|
|
|
|
};
|
2004-01-19 18:21:51 +00:00
|
|
|
/* From Extended Window Manager Hints, draft 1.3:
|
|
|
|
*
|
|
|
|
* _NET_SUPPORTED, ATOM[]/32
|
|
|
|
*
|
|
|
|
* This property MUST be set by the Window Manager
|
|
|
|
* to indicate which hints it supports. For
|
|
|
|
* example: considering _NET_WM_STATE both this
|
|
|
|
* atom and all supported states
|
|
|
|
* e.g. _NET_WM_STATE_MODAL, _NET_WM_STATE_STICKY,
|
|
|
|
* would be listed. This assumes that backwards
|
|
|
|
* incompatible changes will not be made to the
|
|
|
|
* hints (without being renamed).
|
|
|
|
*/
|
2003-05-19 22:43:48 +00:00
|
|
|
screen.rootWindow().changeProperty(m_net_supported, XA_ATOM, 32,
|
|
|
|
PropModeReplace,
|
|
|
|
(unsigned char *) &atomsupported,
|
|
|
|
(sizeof atomsupported)/sizeof atomsupported[0]);
|
2002-10-02 16:26:05 +00:00
|
|
|
|
2004-01-19 18:21:51 +00:00
|
|
|
// update atoms
|
|
|
|
|
|
|
|
updateWorkspaceCount(screen);
|
|
|
|
updateCurrentWorkspace(screen);
|
|
|
|
updateWorkspaceNames(screen);
|
|
|
|
updateClientList(screen);
|
|
|
|
updateViewPort(screen);
|
|
|
|
updateGeometry(screen);
|
|
|
|
updateWorkarea(screen);
|
|
|
|
|
2002-10-02 16:26:05 +00:00
|
|
|
}
|
2002-10-16 19:03:57 +00:00
|
|
|
|
2003-07-28 15:06:36 +00:00
|
|
|
void Ewmh::setupClient(WinClient &winclient) {
|
|
|
|
updateStrut(winclient);
|
|
|
|
}
|
|
|
|
|
2003-07-04 01:03:41 +00:00
|
|
|
void Ewmh::setupFrame(FluxboxWindow &win) {
|
2002-12-01 13:42:15 +00:00
|
|
|
Atom ret_type;
|
|
|
|
int fmt;
|
|
|
|
unsigned long nitems, bytes_after;
|
2004-01-18 19:14:08 +00:00
|
|
|
unsigned char *data = 0;
|
|
|
|
|
2004-01-21 09:37:10 +00:00
|
|
|
/* From Extended Window Manager Hints, draft 1.3:
|
|
|
|
*
|
|
|
|
* _NET_WM_WINDOW_TYPE, ATOM[]/32
|
|
|
|
*
|
|
|
|
* This SHOULD be set by the Client before mapping to a list of atoms
|
|
|
|
* indicating the functional type of the window. This property SHOULD
|
|
|
|
* be used by the window manager in determining the decoration,
|
|
|
|
* stacking position and other behavior of the window. The Client
|
|
|
|
* SHOULD specify window types in order of preference (the first being
|
|
|
|
* most preferable) but MUST include at least one of the basic window
|
|
|
|
* type atoms from the list below. This is to allow for extension of
|
|
|
|
* the list of types whilst providing default behavior for Window
|
|
|
|
* Managers that do not recognize the extensions.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
win.winClient().property(m_net_wm_window_type, 0, 0x7fffffff, False, XA_ATOM,
|
|
|
|
&ret_type, &fmt, &nitems, &bytes_after,
|
|
|
|
&data);
|
|
|
|
if (data) {
|
|
|
|
Atom *atoms = (unsigned long *)data;
|
|
|
|
for (unsigned long l=0; l<nitems; ++l) {
|
|
|
|
/* From Extended Window Manager Hints, draft 1.3:
|
|
|
|
*
|
|
|
|
* _NET_WM_WINDOW_TYPE_DOCK indicates a dock or panel feature.
|
|
|
|
* Typically a Window Manager would keep such windows on top
|
|
|
|
* of all other windows.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
if (atoms[l] == m_net_wm_window_type_dock) {
|
|
|
|
// we also assume it shouldn't be visible in any toolbar
|
2004-02-20 09:07:27 +00:00
|
|
|
win.setFocusHidden(true);
|
|
|
|
win.setIconHidden(true);
|
2004-01-21 09:37:10 +00:00
|
|
|
break;
|
2004-06-20 04:49:33 +00:00
|
|
|
} else if (atoms[l] == m_net_wm_window_type_desktop) {
|
|
|
|
/*
|
|
|
|
* _NET_WM_WINDOW_TYPE_DOCK indicates a "false desktop" window
|
|
|
|
* We let it be the size it wants, but it gets no decoration,
|
|
|
|
* is hidden in the toolbar and window cycling list, plus
|
|
|
|
* windows don't tab with it and is right on the bottom.
|
|
|
|
*/
|
|
|
|
|
|
|
|
win.setFocusHidden(true);
|
|
|
|
win.setIconHidden(true);
|
|
|
|
win.moveToLayer(Fluxbox::instance()->getDesktopLayer());
|
|
|
|
win.setDecorationMask(0);
|
|
|
|
win.setTabable(false);
|
|
|
|
win.setMovable(false);
|
|
|
|
win.setResizable(false);
|
|
|
|
win.stick();
|
2004-01-21 09:37:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
2004-01-21 15:42:14 +00:00
|
|
|
|
|
|
|
setupState(win);
|
|
|
|
|
|
|
|
if (win.winClient().property(m_net_wm_desktop, 0, 1, False, XA_CARDINAL,
|
|
|
|
&ret_type, &fmt, &nitems, &bytes_after,
|
|
|
|
(unsigned char **) &data) && data) {
|
|
|
|
unsigned int desktop = static_cast<unsigned int>(*data);
|
|
|
|
if (desktop == 0xFFFFFFFF && !win.isStuck())
|
|
|
|
win.stick();
|
|
|
|
else
|
|
|
|
win.screen().sendToWorkspace(desktop, &win, false);
|
|
|
|
|
|
|
|
XFree(data);
|
|
|
|
}
|
|
|
|
|
|
|
|
updateWorkspace(win);
|
|
|
|
|
|
|
|
|
2002-10-02 16:26:05 +00:00
|
|
|
}
|
|
|
|
|
2003-09-23 13:52:05 +00:00
|
|
|
void Ewmh::updateFrameClose(FluxboxWindow &win) {
|
|
|
|
clearState(win);
|
|
|
|
}
|
|
|
|
|
2004-01-19 18:21:51 +00:00
|
|
|
void Ewmh::updateFocusedWindow(BScreen &screen, Window win) {
|
|
|
|
/* From Extended Window Manager Hints, draft 1.3:
|
|
|
|
*
|
|
|
|
* _NET_ACTIVE_WINDOW, WINDOW/32
|
|
|
|
*
|
|
|
|
* The window ID of the currently active window or None
|
|
|
|
* if no window has the focus. This is a read-only
|
|
|
|
* property set by the Window Manager.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
screen.rootWindow().changeProperty(m_net_active_window,
|
|
|
|
XA_WINDOW, 32,
|
|
|
|
PropModeReplace,
|
|
|
|
(unsigned char *)&win, 1);
|
|
|
|
}
|
|
|
|
|
2003-03-03 21:51:13 +00:00
|
|
|
void Ewmh::updateClientList(BScreen &screen) {
|
2002-12-01 13:42:15 +00:00
|
|
|
size_t num=0;
|
|
|
|
|
2003-04-15 00:17:59 +00:00
|
|
|
BScreen::Workspaces::const_iterator workspace_it =
|
|
|
|
screen.getWorkspacesList().begin();
|
2004-01-18 19:14:08 +00:00
|
|
|
const BScreen::Workspaces::const_iterator workspace_it_end =
|
2003-04-15 00:17:59 +00:00
|
|
|
screen.getWorkspacesList().end();
|
2002-12-01 13:42:15 +00:00
|
|
|
for (; workspace_it != workspace_it_end; ++workspace_it) {
|
2003-04-15 00:17:59 +00:00
|
|
|
Workspace::Windows::iterator win_it =
|
2003-05-11 17:06:02 +00:00
|
|
|
(*workspace_it)->windowList().begin();
|
2003-04-15 00:17:59 +00:00
|
|
|
Workspace::Windows::iterator win_it_end =
|
2003-05-11 17:06:02 +00:00
|
|
|
(*workspace_it)->windowList().end();
|
2003-04-15 00:17:59 +00:00
|
|
|
for (; win_it != win_it_end; ++win_it) {
|
|
|
|
num += (*win_it)->numClients();
|
|
|
|
}
|
|
|
|
|
2002-12-01 13:42:15 +00:00
|
|
|
}
|
2003-04-15 12:22:06 +00:00
|
|
|
// and count icons
|
|
|
|
BScreen::Icons::const_iterator icon_it = screen.getIconList().begin();
|
|
|
|
BScreen::Icons::const_iterator icon_it_end = screen.getIconList().end();
|
|
|
|
for (; icon_it != icon_it_end; ++icon_it) {
|
|
|
|
num += (*icon_it)->numClients();
|
|
|
|
}
|
2004-06-07 11:46:05 +00:00
|
|
|
|
2002-12-01 13:42:15 +00:00
|
|
|
Window *wl = new (nothrow) Window[num];
|
|
|
|
if (wl == 0) {
|
2004-06-07 11:46:05 +00:00
|
|
|
_FB_USES_NLS;
|
|
|
|
cerr<<_FBTEXT(Ewmh, OutOfMemoryClientList, "Fatal: Out of memory, can't allocate for EWMH client list", "")<<endl;
|
2002-12-01 13:42:15 +00:00
|
|
|
return;
|
|
|
|
}
|
2003-04-15 12:22:06 +00:00
|
|
|
|
2002-12-01 13:42:15 +00:00
|
|
|
//start the iterator from begining
|
|
|
|
workspace_it = screen.getWorkspacesList().begin();
|
|
|
|
int win=0;
|
|
|
|
for (; workspace_it != workspace_it_end; ++workspace_it) {
|
2002-10-02 16:26:05 +00:00
|
|
|
|
2002-12-01 13:42:15 +00:00
|
|
|
// Fill in array of window ID's
|
2003-04-15 00:17:59 +00:00
|
|
|
Workspace::Windows::const_iterator it =
|
2003-05-11 17:06:02 +00:00
|
|
|
(*workspace_it)->windowList().begin();
|
2003-04-15 00:17:59 +00:00
|
|
|
Workspace::Windows::const_iterator it_end =
|
2003-05-11 17:06:02 +00:00
|
|
|
(*workspace_it)->windowList().end();
|
2002-12-01 13:42:15 +00:00
|
|
|
for (; it != it_end; ++it) {
|
2004-01-18 19:14:08 +00:00
|
|
|
if ((*it)->numClients() == 1) {
|
2003-05-15 11:17:29 +00:00
|
|
|
wl[win++] = (*it)->clientWindow();
|
2004-01-18 19:14:08 +00:00
|
|
|
} else {
|
2003-04-14 12:13:36 +00:00
|
|
|
// add every client in fluxboxwindow to list window list
|
2003-04-15 00:17:59 +00:00
|
|
|
std::list<WinClient *>::iterator client_it =
|
|
|
|
(*it)->clientList().begin();
|
|
|
|
std::list<WinClient *>::iterator client_it_end =
|
|
|
|
(*it)->clientList().end();
|
2003-04-14 12:13:36 +00:00
|
|
|
for (; client_it != client_it_end; ++client_it)
|
|
|
|
wl[win++] = (*client_it)->window();
|
|
|
|
}
|
2002-12-01 13:42:15 +00:00
|
|
|
}
|
|
|
|
}
|
2002-12-27 02:15:08 +00:00
|
|
|
|
|
|
|
// plus iconified windows
|
2003-04-15 12:22:06 +00:00
|
|
|
icon_it = screen.getIconList().begin();
|
|
|
|
for (; icon_it != icon_it_end; ++icon_it) {
|
|
|
|
FluxboxWindow::ClientList::iterator client_it = (*icon_it)->clientList().begin();
|
|
|
|
FluxboxWindow::ClientList::iterator client_it_end = (*icon_it)->clientList().end();
|
|
|
|
for (; client_it != client_it_end; ++client_it)
|
|
|
|
wl[win++] = (*client_it)->window();
|
2002-12-27 02:15:08 +00:00
|
|
|
}
|
2002-12-01 13:42:15 +00:00
|
|
|
//number of windows to show in client list
|
|
|
|
num = win;
|
2004-01-19 18:21:51 +00:00
|
|
|
|
|
|
|
/* From Extended Window Manager Hints, draft 1.3:
|
|
|
|
*
|
|
|
|
* _NET_CLIENT_LIST, WINDOW[]/32
|
|
|
|
* _NET_CLIENT_LIST_STACKING, WINDOW[]/32
|
|
|
|
*
|
|
|
|
* These arrays contain all X Windows managed by
|
|
|
|
* the Window Manager. _NET_CLIENT_LIST has
|
|
|
|
* initial mapping order, starting with the oldest
|
|
|
|
* window. _NET_CLIENT_LIST_STACKING has
|
|
|
|
* bottom-to-top stacking order. These properties
|
|
|
|
* SHOULD be set and updated by the Window
|
|
|
|
* Manager.
|
|
|
|
*/
|
2003-12-18 18:03:23 +00:00
|
|
|
screen.rootWindow().changeProperty(m_net_client_list,
|
2004-01-18 19:14:08 +00:00
|
|
|
XA_WINDOW, 32,
|
2003-05-19 22:43:48 +00:00
|
|
|
PropModeReplace, (unsigned char *)wl, num);
|
2004-01-19 18:21:51 +00:00
|
|
|
screen.rootWindow().changeProperty(m_net_client_list_stacking,
|
|
|
|
XA_WINDOW, 32,
|
|
|
|
PropModeReplace, (unsigned char *)wl, num);
|
2002-10-02 16:26:05 +00:00
|
|
|
|
2002-12-01 13:42:15 +00:00
|
|
|
delete [] wl;
|
2002-10-02 16:26:05 +00:00
|
|
|
}
|
|
|
|
|
2003-03-03 21:51:13 +00:00
|
|
|
void Ewmh::updateWorkspaceNames(BScreen &screen) {
|
2004-01-19 18:21:51 +00:00
|
|
|
/* From Extended Window Manager Hints, draft 1.3:
|
|
|
|
*
|
|
|
|
* _NET_DESKTOP_NAMES, UTF8_STRING[]
|
|
|
|
*
|
|
|
|
* The names of all virtual desktops.
|
|
|
|
* This is a list of NULL-terminated strings in UTF-8
|
|
|
|
* encoding [UTF8]. This property MAY be changed by a
|
|
|
|
* Pager or the Window Manager at any time.
|
|
|
|
*
|
|
|
|
* Note: The number of names could be different from
|
|
|
|
* _NET_NUMBER_OF_DESKTOPS. If it is less than
|
|
|
|
* _NET_NUMBER_OF_DESKTOPS, then the desktops with high
|
|
|
|
* numbers are unnamed. If it is larger than
|
|
|
|
* _NET_NUMBER_OF_DESKTOPS, then the excess names outside
|
|
|
|
* of the _NET_NUMBER_OF_DESKTOPS are considered to be
|
|
|
|
* reserved in case the number of desktops is increased.
|
|
|
|
*
|
|
|
|
* Rationale: The name is not a necessary attribute of a
|
|
|
|
* virtual desktop. Thus the availability or unavailability
|
|
|
|
* of names has no impact on virtual desktop functionality.
|
|
|
|
* Since names are set by users and users are likely to
|
|
|
|
* preset names for a fixed number of desktops, it
|
|
|
|
* doesn't make sense to shrink or grow this list when the
|
|
|
|
* number of available desktops changes.
|
|
|
|
*
|
|
|
|
*/
|
2002-12-01 13:42:15 +00:00
|
|
|
XTextProperty text;
|
2003-12-12 15:19:22 +00:00
|
|
|
const BScreen::WorkspaceNames &workspacenames = screen.getWorkspaceNames();
|
|
|
|
const size_t number_of_desks = workspacenames.size();
|
2002-10-02 16:26:05 +00:00
|
|
|
|
2002-12-01 13:42:15 +00:00
|
|
|
char *names[number_of_desks];
|
2002-10-02 16:26:05 +00:00
|
|
|
|
2002-12-01 13:42:15 +00:00
|
|
|
for (size_t i = 0; i < number_of_desks; i++) {
|
2003-12-12 15:19:22 +00:00
|
|
|
names[i] = new char[workspacenames[i].size() + 1]; // +1 for \0
|
|
|
|
memset(names[i], 0, workspacenames[i].size());
|
|
|
|
strcpy(names[i], workspacenames[i].c_str());
|
2002-12-01 13:42:15 +00:00
|
|
|
}
|
2003-12-12 15:19:22 +00:00
|
|
|
|
2002-12-01 13:42:15 +00:00
|
|
|
if (XStringListToTextProperty(names, number_of_desks, &text)) {
|
2003-05-10 22:59:32 +00:00
|
|
|
XSetTextProperty(FbTk::App::instance()->display(), screen.rootWindow().window(),
|
2002-10-02 16:26:05 +00:00
|
|
|
&text, m_net_desktop_names);
|
2002-12-01 13:42:15 +00:00
|
|
|
XFree(text.value);
|
|
|
|
}
|
2004-01-21 09:37:10 +00:00
|
|
|
|
2002-12-01 13:42:15 +00:00
|
|
|
for (size_t i = 0; i < number_of_desks; i++)
|
|
|
|
delete [] names[i];
|
2002-10-02 16:26:05 +00:00
|
|
|
}
|
|
|
|
|
2003-03-03 21:51:13 +00:00
|
|
|
void Ewmh::updateCurrentWorkspace(BScreen &screen) {
|
2004-01-19 18:21:51 +00:00
|
|
|
/* From Extended Window Manager Hints, draft 1.3:
|
|
|
|
*
|
|
|
|
* _NET_CURRENT_DESKTOP desktop, CARDINAL/32
|
|
|
|
*
|
|
|
|
* The index of the current desktop. This is always
|
|
|
|
* an integer between 0 and _NET_NUMBER_OF_DESKTOPS - 1.
|
|
|
|
* This MUST be set and updated by the Window Manager.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
unsigned int workspace = screen.currentWorkspaceID();
|
|
|
|
screen.rootWindow().changeProperty(m_net_current_desktop,
|
|
|
|
XA_CARDINAL, 32,
|
|
|
|
PropModeReplace,
|
2003-05-19 22:43:48 +00:00
|
|
|
(unsigned char *)&workspace, 1);
|
2002-10-02 16:26:05 +00:00
|
|
|
|
|
|
|
}
|
|
|
|
|
2003-03-03 21:51:13 +00:00
|
|
|
void Ewmh::updateWorkspaceCount(BScreen &screen) {
|
2004-01-19 18:21:51 +00:00
|
|
|
/* From Extended Window Manager Hints, draft 1.3:
|
|
|
|
*
|
|
|
|
* _NET_NUMBER_OF_DESKTOPS, CARDINAL/32
|
|
|
|
*
|
|
|
|
* This property SHOULD be set and updated by the
|
|
|
|
* Window Manager to indicate the number of virtual
|
|
|
|
* desktops.
|
|
|
|
*/
|
|
|
|
unsigned int numworkspaces = screen.getCount();
|
|
|
|
screen.rootWindow().changeProperty(m_net_number_of_desktops,
|
|
|
|
XA_CARDINAL, 32,
|
|
|
|
PropModeReplace,
|
2003-05-19 22:43:48 +00:00
|
|
|
(unsigned char *)&numworkspaces, 1);
|
2002-10-02 16:26:05 +00:00
|
|
|
}
|
|
|
|
|
2004-01-19 18:21:51 +00:00
|
|
|
void Ewmh::updateViewPort(BScreen &screen) {
|
|
|
|
/* From Extended Window Manager Hints, draft 1.3:
|
|
|
|
*
|
|
|
|
* _NET_DESKTOP_VIEWPORT x, y, CARDINAL[][2]/32
|
|
|
|
*
|
|
|
|
* Array of pairs of cardinals that define the
|
|
|
|
* top left corner of each desktop's viewport.
|
|
|
|
* For Window Managers that don't support large
|
|
|
|
* desktops, this MUST always be set to (0,0).
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
int value[2] = {0, 0}; // we dont support large desktops
|
|
|
|
screen.rootWindow().changeProperty(m_net_desktop_viewport,
|
|
|
|
XA_CARDINAL, 32,
|
|
|
|
PropModeReplace,
|
|
|
|
(unsigned char *)value, 2);
|
|
|
|
}
|
|
|
|
|
|
|
|
void Ewmh::updateGeometry(BScreen &screen) {
|
|
|
|
/* From Extended Window Manager Hints, draft 1.3:
|
|
|
|
*
|
|
|
|
* _NET_DESKTOP_GEOMETRY width, height, CARDINAL[2]/32
|
|
|
|
*
|
|
|
|
* Array of two cardinals that defines the common size
|
|
|
|
* of all desktops (this is equal to the screen size
|
|
|
|
* if the Window Manager doesn't support large
|
|
|
|
* desktops, otherwise it's equal to the virtual size
|
|
|
|
* of the desktop). This property SHOULD be set by the
|
|
|
|
* Window Manager.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
int value[2] = {screen.width(), screen.height()};
|
|
|
|
screen.rootWindow().changeProperty(m_net_desktop_geometry,
|
|
|
|
XA_CARDINAL, 32,
|
|
|
|
PropModeReplace,
|
|
|
|
(unsigned char *)value, 2);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
void Ewmh::updateWorkarea(BScreen &screen) {
|
|
|
|
/* From Extended Window Manager Hints, draft 1.3:
|
|
|
|
*
|
|
|
|
* _NET_WORKAREA, x, y, width, height CARDINAL[][4]/32
|
|
|
|
*
|
|
|
|
* This property MUST be set by the Window Manager upon
|
|
|
|
* calculating the work area for each desktop. Contains a
|
|
|
|
* geometry for each desktop. These geometries are
|
|
|
|
* specified relative to the viewport on each desktop and
|
|
|
|
* specify an area that is completely contained within the
|
|
|
|
* viewport. Work area SHOULD be used by desktop applications
|
|
|
|
* to place desktop icons appropriately.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
/* !!TODO
|
|
|
|
* Not sure how to handle xinerama stuff here.
|
|
|
|
* So i'm just doing this on the first head.
|
|
|
|
*/
|
|
|
|
unsigned int *coords = new unsigned int[4*screen.getCount()];
|
|
|
|
for (unsigned int i=0; i<screen.getCount()*4; i+=4) {
|
|
|
|
// x, y
|
|
|
|
coords[i] = screen.maxLeft(0);
|
|
|
|
coords[i + 1] = screen.maxTop(0);
|
|
|
|
// width, height
|
|
|
|
coords[i + 2] = screen.maxRight(0) - screen.maxLeft(0);
|
|
|
|
coords[i + 3] = screen.maxBottom(0) - screen.maxTop(0);
|
|
|
|
|
|
|
|
}
|
|
|
|
screen.rootWindow().changeProperty(m_net_workarea,
|
|
|
|
XA_CARDINAL, 32,
|
|
|
|
PropModeReplace,
|
|
|
|
(unsigned char *)coords,
|
|
|
|
4*screen.getCount());
|
|
|
|
}
|
|
|
|
|
2002-10-02 16:26:05 +00:00
|
|
|
void Ewmh::updateState(FluxboxWindow &win) {
|
2003-04-14 12:13:36 +00:00
|
|
|
//!! TODO
|
2002-10-02 16:26:05 +00:00
|
|
|
}
|
|
|
|
|
2003-02-02 16:32:41 +00:00
|
|
|
void Ewmh::updateLayer(FluxboxWindow &win) {
|
2003-04-14 12:13:36 +00:00
|
|
|
//!! TODO _NET_WM_WINDOW_TYPE
|
2003-02-02 16:32:41 +00:00
|
|
|
}
|
|
|
|
|
2002-10-02 16:26:05 +00:00
|
|
|
void Ewmh::updateHints(FluxboxWindow &win) {
|
|
|
|
}
|
|
|
|
|
|
|
|
void Ewmh::updateWorkspace(FluxboxWindow &win) {
|
2003-05-15 11:17:29 +00:00
|
|
|
int workspace = win.workspaceNumber();
|
2002-12-01 13:42:15 +00:00
|
|
|
if (win.isStuck())
|
|
|
|
workspace = 0xFFFFFFFF; // appear on all desktops/workspaces
|
2002-10-02 16:26:05 +00:00
|
|
|
|
2003-07-02 14:31:43 +00:00
|
|
|
FluxboxWindow::ClientList::iterator it = win.clientList().begin();
|
|
|
|
FluxboxWindow::ClientList::iterator it_end = win.clientList().end();
|
|
|
|
for (; it != it_end; ++it) {
|
|
|
|
(*it)->changeProperty(m_net_wm_desktop, XA_CARDINAL, 32, PropModeReplace,
|
|
|
|
(unsigned char *)&workspace, 1);
|
|
|
|
}
|
|
|
|
|
2002-10-02 16:26:05 +00:00
|
|
|
}
|
|
|
|
|
2004-01-21 09:37:10 +00:00
|
|
|
|
2002-10-16 19:03:57 +00:00
|
|
|
// return true if we did handle the atom here
|
2004-02-14 12:15:35 +00:00
|
|
|
bool Ewmh::checkClientMessage(const XClientMessageEvent &ce,
|
|
|
|
BScreen * screen, WinClient * const winclient) {
|
2002-12-01 13:42:15 +00:00
|
|
|
if (ce.message_type == m_net_wm_desktop) {
|
|
|
|
// ce.data.l[0] = workspace number
|
2004-03-07 23:37:39 +00:00
|
|
|
// valid window
|
|
|
|
|
|
|
|
if (winclient == 0 || winclient->fbwindow() == 0)
|
2002-12-01 13:42:15 +00:00
|
|
|
return true;
|
2004-03-07 23:37:39 +00:00
|
|
|
|
|
|
|
FluxboxWindow *fbwin = winclient->fbwindow();
|
|
|
|
|
|
|
|
// if it's stick, make sure it is stuck.
|
|
|
|
// otherwise, make sure it isn't stuck
|
|
|
|
if (ce.data.l[0] == 0xFFFFFFFF) {
|
|
|
|
if (!fbwin->isStuck())
|
|
|
|
fbwin->stick();
|
|
|
|
return true;
|
|
|
|
} else if (fbwin->isStuck())
|
|
|
|
fbwin->stick();
|
|
|
|
|
|
|
|
// the screen is the root window of the message,
|
|
|
|
// which doesn't apply here (so borrow the variable :) )
|
|
|
|
screen = &fbwin->screen();
|
|
|
|
// valid workspace number?
|
|
|
|
if (static_cast<unsigned int>(ce.data.l[0]) < screen->getCount())
|
|
|
|
screen->sendToWorkspace(ce.data.l[0], fbwin, false);
|
|
|
|
|
2002-12-01 13:42:15 +00:00
|
|
|
return true;
|
|
|
|
} else if (ce.message_type == m_net_wm_state) {
|
2003-07-28 15:06:36 +00:00
|
|
|
if (winclient == 0 || winclient->fbwindow() == 0)
|
2002-12-01 13:42:15 +00:00
|
|
|
return true;
|
2003-07-28 15:06:36 +00:00
|
|
|
|
|
|
|
FluxboxWindow &win = *winclient->fbwindow();
|
2002-12-01 13:42:15 +00:00
|
|
|
// ce.data.l[0] = the action (remove, add or toggle)
|
|
|
|
// ce.data.l[1] = the first property to alter
|
|
|
|
// ce.data.l[2] = second property to alter (can be zero)
|
|
|
|
if (ce.data.l[0] == STATE_REMOVE) {
|
2003-07-28 15:06:36 +00:00
|
|
|
setState(win, ce.data.l[1], false);
|
|
|
|
setState(win, ce.data.l[2], false);
|
2002-12-01 13:42:15 +00:00
|
|
|
} else if (ce.data.l[0] == STATE_ADD) {
|
2003-07-28 15:06:36 +00:00
|
|
|
setState(win, ce.data.l[1], true);
|
|
|
|
setState(win, ce.data.l[2], true);
|
2002-12-01 13:42:15 +00:00
|
|
|
} else if (ce.data.l[0] == STATE_TOGGLE) {
|
2003-07-28 15:06:36 +00:00
|
|
|
toggleState(win, ce.data.l[1]);
|
|
|
|
toggleState(win, ce.data.l[2]);
|
2002-12-01 13:42:15 +00:00
|
|
|
}
|
2003-06-18 13:34:30 +00:00
|
|
|
|
2002-12-01 13:42:15 +00:00
|
|
|
return true;
|
|
|
|
} else if (ce.message_type == m_net_number_of_desktops) {
|
|
|
|
if (screen == 0)
|
|
|
|
return true;
|
|
|
|
// ce.data.l[0] = number of workspaces
|
2002-10-16 19:03:57 +00:00
|
|
|
|
2002-12-01 13:42:15 +00:00
|
|
|
// no need to alter number of desktops if they are the same
|
|
|
|
// or if requested number of workspace is less than zero
|
|
|
|
if (screen->getCount() == static_cast<unsigned int>(ce.data.l[0]) ||
|
|
|
|
ce.data.l[0] < 0)
|
|
|
|
return true;
|
|
|
|
|
|
|
|
if (screen->getCount() > static_cast<unsigned int>(ce.data.l[0])) {
|
|
|
|
// remove last workspace until we have
|
|
|
|
// the same number of workspaces
|
|
|
|
while (screen->getCount() != static_cast<unsigned int>(ce.data.l[0])) {
|
|
|
|
screen->removeLastWorkspace();
|
|
|
|
if (screen->getCount() == 1) // must have at least one workspace
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
} else { // add workspaces to screen until workspace count match the requested size
|
|
|
|
while (screen->getCount() != static_cast<unsigned int>(ce.data.l[0])) {
|
|
|
|
screen->addWorkspace();
|
|
|
|
}
|
|
|
|
}
|
2002-10-16 19:03:57 +00:00
|
|
|
|
2002-12-01 13:42:15 +00:00
|
|
|
return true;
|
|
|
|
} else if (ce.message_type == m_net_current_desktop) {
|
|
|
|
if (screen == 0)
|
|
|
|
return true;
|
|
|
|
// ce.data.l[0] = workspace number
|
2002-10-16 19:03:57 +00:00
|
|
|
|
2002-12-01 13:42:15 +00:00
|
|
|
// prevent out of range value
|
|
|
|
if (static_cast<unsigned int>(ce.data.l[0]) >= screen->getCount())
|
|
|
|
return true;
|
|
|
|
screen->changeWorkspaceID(ce.data.l[0]);
|
|
|
|
return true;
|
|
|
|
} else if (ce.message_type == m_net_active_window) {
|
2002-10-16 19:03:57 +00:00
|
|
|
|
2002-12-01 13:42:15 +00:00
|
|
|
// make sure we have a valid window
|
2003-07-28 15:06:36 +00:00
|
|
|
if (winclient == 0)
|
2002-12-01 13:42:15 +00:00
|
|
|
return true;
|
|
|
|
// ce.window = window to focus
|
2002-10-16 19:03:57 +00:00
|
|
|
|
2003-07-28 15:06:36 +00:00
|
|
|
winclient->focus();
|
2004-01-19 18:21:51 +00:00
|
|
|
if (winclient->fbwindow())
|
|
|
|
winclient->fbwindow()->raise();
|
|
|
|
|
2002-12-01 13:42:15 +00:00
|
|
|
return true;
|
|
|
|
} else if (ce.message_type == m_net_close_window) {
|
2003-07-28 15:06:36 +00:00
|
|
|
if (winclient == 0)
|
2002-12-01 13:42:15 +00:00
|
|
|
return true;
|
|
|
|
// ce.window = window to close (which in this case is the win argument)
|
2003-07-28 15:06:36 +00:00
|
|
|
winclient->sendClose();
|
2002-12-01 13:42:15 +00:00
|
|
|
return true;
|
|
|
|
} else if (ce.message_type == m_net_moveresize_window) {
|
2004-02-14 12:15:35 +00:00
|
|
|
if (winclient == 0 || winclient->fbwindow() == 0)
|
2002-12-01 13:42:15 +00:00
|
|
|
return true;
|
|
|
|
// ce.data.l[0] = gravity and flags
|
|
|
|
// ce.data.l[1] = x
|
|
|
|
// ce.data.l[2] = y
|
|
|
|
// ce.data.l[3] = width
|
|
|
|
// ce.data.l[4] = height
|
|
|
|
// TODO: gravity and flags
|
2003-07-28 15:06:36 +00:00
|
|
|
winclient->fbwindow()->moveResize(ce.data.l[1], ce.data.l[2],
|
2004-01-18 19:14:08 +00:00
|
|
|
ce.data.l[3], ce.data.l[4]);
|
2002-12-01 13:42:15 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
// we didn't handle the ce.message_type here
|
|
|
|
return false;
|
2002-10-02 16:26:05 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2003-07-28 15:06:36 +00:00
|
|
|
bool Ewmh::propertyNotify(WinClient &winclient, Atom the_property) {
|
2003-06-18 13:34:30 +00:00
|
|
|
if (the_property == m_net_wm_strut) {
|
2003-07-28 15:06:36 +00:00
|
|
|
updateStrut(winclient);
|
2003-06-18 13:34:30 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2002-10-02 16:26:05 +00:00
|
|
|
void Ewmh::createAtoms() {
|
2003-08-11 14:51:15 +00:00
|
|
|
|
2003-03-04 11:06:03 +00:00
|
|
|
Display *disp = FbTk::App::instance()->display();
|
2003-08-11 14:51:15 +00:00
|
|
|
|
2002-12-01 13:42:15 +00:00
|
|
|
m_net_supported = XInternAtom(disp, "_NET_SUPPORTED", False);
|
|
|
|
m_net_client_list = XInternAtom(disp, "_NET_CLIENT_LIST", False);
|
|
|
|
m_net_client_list_stacking = XInternAtom(disp, "_NET_CLIENT_LIST_STACKING", False);
|
|
|
|
m_net_number_of_desktops = XInternAtom(disp, "_NET_NUMBER_OF_DESKTOPS", False);
|
|
|
|
m_net_desktop_geometry = XInternAtom(disp, "_NET_DESKTOP_GEOMETRY", False);
|
|
|
|
m_net_desktop_viewport = XInternAtom(disp, "_NET_DESKTOP_VIEWPORT", False);
|
|
|
|
m_net_current_desktop = XInternAtom(disp, "_NET_CURRENT_DESKTOP", False);
|
|
|
|
m_net_desktop_names = XInternAtom(disp, "_NET_DESKTOP_NAMES", False);
|
|
|
|
m_net_active_window = XInternAtom(disp, "_NET_ACTIVE_WINDOW", False);
|
|
|
|
m_net_workarea = XInternAtom(disp, "_NET_WORKAREA", False);
|
|
|
|
m_net_supporting_wm_check = XInternAtom(disp, "_NET_SUPPORTING_WM_CHECK", False);
|
|
|
|
m_net_virtual_roots = XInternAtom(disp, "_NET_VIRTUAL_ROOTS", False);
|
|
|
|
|
|
|
|
m_net_close_window = XInternAtom(disp, "_NET_CLOSE_WINDOW", False);
|
|
|
|
m_net_moveresize_window = XInternAtom(disp, "_NET_MOVERESIZE_WINDOW", False);
|
2002-11-23 14:54:50 +00:00
|
|
|
|
2002-12-01 13:42:15 +00:00
|
|
|
// TODO: implement this one
|
|
|
|
m_net_wm_moveresize = XInternAtom(disp, "_NET_WM_MOVERESIZE", False);
|
2002-10-02 16:26:05 +00:00
|
|
|
|
2002-12-01 13:42:15 +00:00
|
|
|
m_net_properties = XInternAtom(disp, "_NET_PROPERTIES", False);
|
|
|
|
m_net_wm_name = XInternAtom(disp, "_NET_WM_NAME", False);
|
|
|
|
m_net_wm_desktop = XInternAtom(disp, "_NET_WM_DESKTOP", False);
|
2004-01-21 09:37:10 +00:00
|
|
|
|
|
|
|
// type atoms
|
2002-12-01 13:42:15 +00:00
|
|
|
m_net_wm_window_type = XInternAtom(disp, "_NET_WM_WINDOW_TYPE", False);
|
2004-01-21 09:37:10 +00:00
|
|
|
m_net_wm_window_type_dock = XInternAtom(disp, "_NET_WM_WINDOW_TYPE_DOCK", False);
|
2004-06-20 04:49:33 +00:00
|
|
|
m_net_wm_window_type_desktop = XInternAtom(disp, "_NET_WM_WINDOW_TYPE_DESKTOP", False);
|
2004-01-21 09:37:10 +00:00
|
|
|
|
2002-12-01 13:42:15 +00:00
|
|
|
// state atom and the supported state atoms
|
|
|
|
m_net_wm_state = XInternAtom(disp, "_NET_WM_STATE", False);
|
|
|
|
m_net_wm_state_sticky = XInternAtom(disp, "_NET_WM_STATE_STICKY", False);
|
|
|
|
m_net_wm_state_shaded = XInternAtom(disp, "_NET_WM_STATE_SHADED", False);
|
2003-08-27 21:06:04 +00:00
|
|
|
m_net_wm_state_maximized_horz = XInternAtom(disp, "_NET_WM_STATE_MAXIMIZED_HORZ", False);
|
|
|
|
m_net_wm_state_maximized_vert = XInternAtom(disp, "_NET_WM_STATE_MAXIMIZED_VERT", False);
|
2003-09-23 13:52:05 +00:00
|
|
|
m_net_wm_state_fullscreen = XInternAtom(disp, "_NET_WM_STATE_FULLSCREEN", False);
|
2004-01-18 19:14:08 +00:00
|
|
|
m_net_wm_state_hidden = XInternAtom(disp, "_NET_WM_STATE_HIDDEN", False);
|
|
|
|
m_net_wm_state_skip_taskbar = XInternAtom(disp, "_NET_WM_STATE_SKIP_TASKBAR", False);
|
2004-01-19 22:43:08 +00:00
|
|
|
m_net_wm_state_above = XInternAtom(disp, "_NET_WM_STATE_ABOVE", False);
|
|
|
|
m_net_wm_state_below = XInternAtom(disp, "_NET_WM_STATE_BELOW", False);
|
|
|
|
|
2004-01-18 19:14:08 +00:00
|
|
|
|
2002-12-01 13:42:15 +00:00
|
|
|
m_net_wm_strut = XInternAtom(disp, "_NET_WM_STRUT", False);
|
|
|
|
m_net_wm_icon_geometry = XInternAtom(disp, "_NET_WM_ICON_GEOMETRY", False);
|
|
|
|
m_net_wm_icon = XInternAtom(disp, "_NET_WM_ICON", False);
|
|
|
|
m_net_wm_pid = XInternAtom(disp, "_NET_WM_PID", False);
|
|
|
|
m_net_wm_handled_icons = XInternAtom(disp, "_NET_WM_HANDLED_ICONS", False);
|
|
|
|
|
|
|
|
m_net_wm_ping = XInternAtom(disp, "_NET_WM_PING", False);
|
2002-10-02 16:26:05 +00:00
|
|
|
}
|
|
|
|
|
2003-09-23 13:52:05 +00:00
|
|
|
|
|
|
|
void Ewmh::setFullscreen(FluxboxWindow &win, bool value) {
|
|
|
|
// fullscreen implies maximised, above dock layer,
|
|
|
|
// and no decorations (or decorations offscreen)
|
|
|
|
WindowState *saved_state = getState(win);
|
|
|
|
if (value) {
|
|
|
|
// fullscreen on
|
|
|
|
if (!saved_state) { // not already fullscreen
|
|
|
|
saved_state = new WindowState(win.x(), win.y(), win.width(),
|
|
|
|
win.height(), win.layerNum(), win.decorationMask());
|
|
|
|
saveState(win, saved_state);
|
|
|
|
|
|
|
|
// actually make it fullscreen
|
|
|
|
|
|
|
|
// clear decorations
|
|
|
|
win.setDecorationMask(0);
|
|
|
|
|
|
|
|
// be xinerama aware
|
|
|
|
BScreen &screen = win.screen();
|
|
|
|
int head = screen.getHead(win.fbWindow());
|
|
|
|
win.moveResize(screen.getHeadX(head), screen.getHeadY(head),
|
|
|
|
screen.getHeadWidth(head), screen.getHeadHeight(head));
|
|
|
|
win.moveToLayer(Fluxbox::instance()->getAboveDockLayer());
|
|
|
|
}
|
|
|
|
} else { // turn off fullscreen
|
|
|
|
if (saved_state) { // no saved state, can't restore it
|
|
|
|
win.setDecorationMask(saved_state->decor);
|
|
|
|
win.moveResize(saved_state->x, saved_state->y,
|
|
|
|
saved_state->width, saved_state->height);
|
|
|
|
win.moveToLayer(saved_state->layer);
|
|
|
|
clearState(win);
|
|
|
|
saved_state = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2002-10-11 10:20:33 +00:00
|
|
|
// set window state
|
2003-09-23 13:52:05 +00:00
|
|
|
void Ewmh::setState(FluxboxWindow &win, Atom state, bool value) {
|
2002-10-11 10:20:33 +00:00
|
|
|
|
2002-12-01 13:42:15 +00:00
|
|
|
if (state == m_net_wm_state_sticky) { // STICKY
|
|
|
|
if (value && !win.isStuck() ||
|
|
|
|
(!value && win.isStuck()))
|
|
|
|
win.stick();
|
|
|
|
} else if (state == m_net_wm_state_shaded) { // SHADED
|
|
|
|
if ((value && !win.isShaded()) ||
|
|
|
|
(!value && win.isShaded()))
|
|
|
|
win.shade();
|
2003-08-27 21:06:04 +00:00
|
|
|
} else if (state == m_net_wm_state_maximized_horz ) { // maximized Horizontal
|
|
|
|
if ((value && !win.isMaximized()) ||
|
|
|
|
(!value && win.isMaximized()))
|
|
|
|
win.maximizeHorizontal();
|
|
|
|
} else if (state == m_net_wm_state_maximized_vert) { // maximized Vertical
|
|
|
|
if ((value && !win.isMaximized()) ||
|
|
|
|
(!value && win.isMaximized()))
|
|
|
|
win.maximizeVertical();
|
2003-09-23 13:52:05 +00:00
|
|
|
} else if (state == m_net_wm_state_fullscreen) { // fullscreen
|
|
|
|
setFullscreen(win, value);
|
2004-01-18 19:14:08 +00:00
|
|
|
} else if (state == m_net_wm_state_hidden ||
|
2004-01-19 22:43:08 +00:00
|
|
|
state == m_net_wm_state_skip_taskbar) {
|
2004-02-20 09:07:27 +00:00
|
|
|
win.setFocusHidden(value);
|
|
|
|
win.setIconHidden(win.isFocusHidden());
|
2004-01-19 22:43:08 +00:00
|
|
|
} else if (state == m_net_wm_state_below) {
|
|
|
|
if (value)
|
|
|
|
win.moveToLayer(Fluxbox::instance()->getBottomLayer());
|
|
|
|
else
|
|
|
|
win.moveToLayer(Fluxbox::instance()->getNormalLayer());
|
|
|
|
|
|
|
|
} else if (state == m_net_wm_state_above) {
|
|
|
|
if (value)
|
|
|
|
win.moveToLayer(Fluxbox::instance()->getAboveDockLayer());
|
|
|
|
else
|
|
|
|
win.moveToLayer(Fluxbox::instance()->getNormalLayer());
|
|
|
|
|
|
|
|
}
|
2002-10-11 10:20:33 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// toggle window state
|
2003-09-23 13:52:05 +00:00
|
|
|
void Ewmh::toggleState(FluxboxWindow &win, Atom state) {
|
2002-12-01 13:42:15 +00:00
|
|
|
if (state == m_net_wm_state_sticky) {
|
|
|
|
win.stick();
|
2003-08-27 21:06:04 +00:00
|
|
|
} else if (state == m_net_wm_state_shaded){
|
2002-12-01 13:42:15 +00:00
|
|
|
win.shade();
|
2003-08-27 21:06:04 +00:00
|
|
|
} else if (state == m_net_wm_state_maximized_horz ) { // maximized Horizontal
|
|
|
|
win.maximizeHorizontal();
|
|
|
|
} else if (state == m_net_wm_state_maximized_vert) { // maximized Vertical
|
|
|
|
win.maximizeVertical();
|
2003-09-23 13:52:05 +00:00
|
|
|
} else if (state == m_net_wm_state_fullscreen) { // fullscreen
|
|
|
|
setFullscreen(win, getState(win) == 0); // toggle current state
|
2004-01-18 19:14:08 +00:00
|
|
|
} else if (state == m_net_wm_state_hidden ||
|
|
|
|
state == m_net_wm_state_skip_taskbar) {
|
2004-02-20 09:07:27 +00:00
|
|
|
win.setFocusHidden(!win.isFocusHidden());
|
|
|
|
win.setIconHidden(!win.isIconHidden());
|
2004-01-19 22:43:08 +00:00
|
|
|
} else if (state == m_net_wm_state_below) {
|
|
|
|
if (win.layerNum() == Fluxbox::instance()->getBottomLayer())
|
|
|
|
win.moveToLayer(Fluxbox::instance()->getNormalLayer());
|
|
|
|
else
|
|
|
|
win.moveToLayer(Fluxbox::instance()->getBottomLayer());
|
2004-01-21 15:42:14 +00:00
|
|
|
|
2004-01-19 22:43:08 +00:00
|
|
|
} else if (state == m_net_wm_state_above) {
|
|
|
|
if (win.layerNum() == Fluxbox::instance()->getAboveDockLayer())
|
|
|
|
win.moveToLayer(Fluxbox::instance()->getNormalLayer());
|
|
|
|
else
|
2004-01-21 15:42:14 +00:00
|
|
|
win.moveToLayer(Fluxbox::instance()->getAboveDockLayer());
|
2003-08-27 21:06:04 +00:00
|
|
|
}
|
2004-01-19 22:43:08 +00:00
|
|
|
|
2002-10-11 10:20:33 +00:00
|
|
|
}
|
|
|
|
|
2003-06-18 13:34:30 +00:00
|
|
|
|
2003-07-28 15:06:36 +00:00
|
|
|
void Ewmh::updateStrut(WinClient &winclient) {
|
2003-06-18 13:34:30 +00:00
|
|
|
Atom ret_type = 0;
|
|
|
|
int fmt = 0;
|
|
|
|
unsigned long nitems = 0, bytes_after = 0;
|
|
|
|
long *data = 0;
|
2003-07-28 15:06:36 +00:00
|
|
|
if (winclient.property(m_net_wm_strut, 0, 4, False, XA_CARDINAL,
|
2003-06-18 13:34:30 +00:00
|
|
|
&ret_type, &fmt, &nitems, &bytes_after,
|
|
|
|
(unsigned char **) &data) && data) {
|
2004-01-21 15:42:14 +00:00
|
|
|
|
|
|
|
winclient.setStrut(winclient.screen().requestStrut(data[0], data[1],
|
|
|
|
data[2], data[3]));
|
2003-07-28 15:06:36 +00:00
|
|
|
winclient.screen().updateAvailableWorkspaceArea();
|
2003-06-18 13:34:30 +00:00
|
|
|
}
|
|
|
|
}
|
2003-09-23 13:52:05 +00:00
|
|
|
|
2004-01-21 15:42:14 +00:00
|
|
|
void Ewmh::setupState(FluxboxWindow &win) {
|
|
|
|
/* From Extended Window Manager Hints, draft 1.3:
|
|
|
|
*
|
|
|
|
* _NET_WM_STATE, ATOM[]
|
|
|
|
*
|
|
|
|
* A list of hints describing the window state. Atoms present in
|
|
|
|
* the list MUST be considered set, atoms not present in the list
|
|
|
|
* MUST be considered not set. The Window Manager SHOULD honor
|
|
|
|
* _NET_WM_STATE whenever a withdrawn window requests to be mapped.
|
|
|
|
* A Client wishing to change the state of a window MUST send a
|
|
|
|
* _NET_WM_STATE client message to the root window (see below).
|
|
|
|
* The Window Manager MUST keep this property updated to reflect
|
|
|
|
* the current state of the window.
|
|
|
|
*
|
|
|
|
* The Window Manager should remove the property whenever a window
|
|
|
|
* is withdrawn, but it should leave the property in place when it
|
|
|
|
* is shutting down, e.g. in response to losing ownership of the
|
|
|
|
* WM_Sn manager selection.
|
|
|
|
*/
|
|
|
|
Atom ret_type;
|
|
|
|
int fmt;
|
|
|
|
unsigned long nitems, bytes_after;
|
|
|
|
unsigned char *data = 0;
|
|
|
|
|
|
|
|
win.winClient().property(m_net_wm_state, 0, 0x7fffffff, False, XA_ATOM,
|
|
|
|
&ret_type, &fmt, &nitems, &bytes_after,
|
|
|
|
&data);
|
|
|
|
if (data) {
|
|
|
|
Atom *states = (Atom *)data;
|
|
|
|
for (unsigned long i=0; i < nitems; ++i)
|
|
|
|
setState(win, states[i], true);
|
|
|
|
|
|
|
|
XFree(data);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2003-09-23 13:52:05 +00:00
|
|
|
Ewmh::WindowState::WindowState(int t_x, int t_y,
|
|
|
|
unsigned int t_width,
|
|
|
|
unsigned int t_height,
|
|
|
|
int t_layer, unsigned int t_decor) :
|
|
|
|
x(t_x), y(t_y),
|
|
|
|
layer(t_layer),
|
|
|
|
width(t_width),
|
|
|
|
height(t_height),
|
|
|
|
decor(t_decor)
|
|
|
|
{}
|
|
|
|
|
|
|
|
Ewmh::WindowState *Ewmh::getState(FluxboxWindow &win) {
|
|
|
|
SavedState::iterator it = m_savedstate.find(&win);
|
|
|
|
if (it == m_savedstate.end())
|
|
|
|
return 0;
|
|
|
|
else
|
|
|
|
return it->second;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Ewmh::clearState(FluxboxWindow &win) {
|
|
|
|
WindowState *state = 0;
|
|
|
|
SavedState::iterator it = m_savedstate.find(&win);
|
|
|
|
if (it == m_savedstate.end())
|
|
|
|
return;
|
|
|
|
|
|
|
|
state = it->second;
|
|
|
|
|
|
|
|
m_savedstate.erase(it);
|
|
|
|
delete state;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Ewmh::saveState(FluxboxWindow &win, WindowState *state) {
|
|
|
|
m_savedstate[&win] = state;
|
|
|
|
}
|