fluxbox/src/Ewmh.cc

388 lines
14 KiB
C++
Raw Normal View History

2002-10-02 16:26:05 +00:00
// Ewmh.cc for fluxbox
// Copyright (c) 2002 Henrik Kinnunen (fluxgen@fluxbox.org)
//
// 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,
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
// 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.
// $Id: Ewmh.cc,v 1.9 2003/01/05 23:04:46 fluxgen Exp $
2002-10-02 16:26:05 +00:00
#include "Ewmh.hh"
#include "Screen.hh"
#include "Window.hh"
2002-10-16 19:03:57 +00:00
#include "fluxbox.hh"
2002-10-02 16:26:05 +00:00
#include <iostream>
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()) {
XDestroyWindow(BaseDisplay::getXDisplay(), m_windows.back());
m_windows.pop_back();
}
2002-10-02 16:26:05 +00:00
}
void Ewmh::initForScreen(const BScreen &screen) {
2002-12-01 13:42:15 +00:00
Display *disp = BaseDisplay::getXDisplay();
2002-10-02 16:26:05 +00:00
2002-12-01 13:42:15 +00:00
Window wincheck = XCreateSimpleWindow(disp,
screen.getRootWindow(), 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) {
m_windows.push_back(wincheck);
2002-10-02 16:26:05 +00:00
2002-12-01 13:42:15 +00:00
XChangeProperty(disp, screen.getRootWindow(), m_net_supporting_wm_check, XA_WINDOW, 32,
2002-10-02 16:26:05 +00:00
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
m_net_wm_state,
// states that we support:
m_net_wm_state_sticky,
m_net_wm_state_shaded,
2002-10-02 16:26:05 +00:00
2002-12-01 13:42:15 +00:00
m_net_wm_desktop,
2002-10-16 19:03:57 +00:00
2002-12-01 13:42:15 +00:00
// root properties
m_net_client_list,
m_net_number_of_desktops,
m_net_current_desktop,
m_net_active_window,
m_net_close_window,
m_net_moveresize_window,
m_net_desktop_names,
m_net_supporting_wm_check
};
XChangeProperty(disp, screen.getRootWindow(),
m_net_supported, XA_ATOM, 32,
PropModeReplace, (unsigned char *) &atomsupported, (sizeof atomsupported)/sizeof atomsupported[0]);
2002-10-02 16:26:05 +00:00
}
2002-10-16 19:03:57 +00:00
2002-10-02 16:26:05 +00:00
void Ewmh::setupWindow(FluxboxWindow &win) {
2002-10-16 19:03:57 +00:00
2002-12-01 13:42:15 +00:00
Display *disp = BaseDisplay::getXDisplay();
Atom ret_type;
int fmt;
unsigned long nitems, bytes_after;
long *data = 0;
2002-10-16 19:03:57 +00:00
/*
2002-10-02 16:26:05 +00:00
if (XGetWindowProperty(disp, win.getClientWindow(),
2002-12-01 13:42:15 +00:00
m_net_wm_state, 0, 1, False, XA_CARDINAL,
&ret_type, &fmt, &nitems, &bytes_after,
(unsigned char **) &data) == Success && data) {
flags = *data;
setState(win, flags);
XFree(data);
2002-10-02 16:26:05 +00:00
}
*/
2002-12-01 13:42:15 +00:00
if (XGetWindowProperty(disp, win.getClientWindow(),
m_net_wm_desktop, 0, 1, False, XA_CARDINAL,
&ret_type, &fmt, &nitems, &bytes_after,
(unsigned char **) &data) == Success && data) {
unsigned int desktop = static_cast<unsigned int>(*data);
if (desktop == 0xFFFFFFFF && !win.isStuck())
win.stick();
else if (win.getScreen())
win.getScreen()->sendToWorkspace(desktop, &win, false);
XFree(data);
}
2002-10-02 16:26:05 +00:00
}
void Ewmh::updateClientList(const BScreen &screen) {
2002-12-01 13:42:15 +00:00
size_t num=0;
BScreen::Workspaces::const_iterator workspace_it = screen.getWorkspacesList().begin();
BScreen::Workspaces::const_iterator workspace_it_end = screen.getWorkspacesList().end();
for (; workspace_it != workspace_it_end; ++workspace_it) {
num += (*workspace_it)->getWindowList().size();
}
//int num = getCurrentWorkspace()->getWindowList().size();
2002-10-02 16:26:05 +00:00
2002-12-01 13:42:15 +00:00
Window *wl = new (nothrow) Window[num];
if (wl == 0) {
cerr<<"Fatal: Out of memory, can't allocate for Ewmh client list"<<endl;
return;
}
//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
Workspace::Windows::const_iterator it = (*workspace_it)->getWindowList().begin();
Workspace::Windows::const_iterator it_end = (*workspace_it)->getWindowList().end();
for (; it != it_end; ++it) {
wl[win++] = (*it)->getClientWindow();
}
}
// plus iconified windows
BScreen::Icons::const_iterator it = screen.getIconList().begin();
BScreen::Icons::const_iterator it_end = screen.getIconList().end();
for (; it != it_end; ++it) {
wl[win++] = (*it)->getClientWindow();
}
2002-12-01 13:42:15 +00:00
//number of windows to show in client list
num = win;
XChangeProperty(BaseDisplay::getXDisplay(),
screen.getRootWindow(),
m_net_client_list,
XA_CARDINAL, 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
}
void Ewmh::updateWorkspaceNames(const BScreen &screen) {
2002-12-01 13:42:15 +00:00
XTextProperty text;
const size_t number_of_desks = screen.getWorkspaceNames().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++) {
names[i] = new char[screen.getWorkspaceNames()[i].size()];
strcpy(names[i], screen.getWorkspaceNames()[i].c_str());
}
2002-10-02 16:26:05 +00:00
2002-12-01 13:42:15 +00:00
if (XStringListToTextProperty(names, number_of_desks, &text)) {
XSetTextProperty(BaseDisplay::getXDisplay(), screen.getRootWindow(),
2002-10-02 16:26:05 +00:00
&text, m_net_desktop_names);
2002-12-01 13:42:15 +00:00
XFree(text.value);
}
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++)
delete [] names[i];
2002-10-02 16:26:05 +00:00
}
void Ewmh::updateCurrentWorkspace(const BScreen &screen) {
2002-12-01 13:42:15 +00:00
size_t workspace = screen.getCurrentWorkspaceID();
XChangeProperty(BaseDisplay::getXDisplay(),
screen.getRootWindow(),
m_net_current_desktop, XA_CARDINAL, 32, PropModeReplace,
(unsigned char *)&workspace, 1);
2002-10-02 16:26:05 +00:00
}
void Ewmh::updateWorkspaceCount(const BScreen &screen) {
2002-12-01 13:42:15 +00:00
size_t numworkspaces = screen.getCount();
XChangeProperty(BaseDisplay::getXDisplay(), screen.getRootWindow(),
m_net_number_of_desktops, XA_CARDINAL, 32, PropModeReplace,
(unsigned char *)&numworkspaces, 1);
2002-10-02 16:26:05 +00:00
}
void Ewmh::updateState(FluxboxWindow &win) {
}
void Ewmh::updateHints(FluxboxWindow &win) {
}
void Ewmh::updateWorkspace(FluxboxWindow &win) {
2002-12-01 13:42:15 +00:00
int workspace = win.getWorkspaceNumber();
if (win.isStuck())
workspace = 0xFFFFFFFF; // appear on all desktops/workspaces
2002-10-02 16:26:05 +00:00
2002-12-01 13:42:15 +00:00
XChangeProperty(BaseDisplay::getXDisplay(), win.getClientWindow(),
m_net_wm_desktop, XA_CARDINAL, 32, PropModeReplace,
(unsigned char *)&workspace, 1);
2002-10-02 16:26:05 +00:00
}
2002-10-16 19:03:57 +00:00
// return true if we did handle the atom here
bool Ewmh::checkClientMessage(const XClientMessageEvent &ce, BScreen * const screen, FluxboxWindow * const win) {
2002-12-01 13:42:15 +00:00
if (ce.message_type == m_net_wm_desktop) {
if (screen == 0)
return true;
// ce.data.l[0] = workspace number
// valid window and workspace number?
if (win == 0 || static_cast<unsigned int>(ce.data.l[0]) >= screen->getCount())
return true;
2002-10-16 19:03:57 +00:00
2002-12-01 13:42:15 +00:00
screen->sendToWorkspace(ce.data.l[0], win, false);
return true;
} else if (ce.message_type == m_net_wm_state) {
if (win == 0)
return true;
// 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) {
setState(*win, ce.data.l[1], false);
setState(*win, ce.data.l[2], false);
} else if (ce.data.l[0] == STATE_ADD) {
setState(*win, ce.data.l[1], true);
setState(*win, ce.data.l[2], true);
} else if (ce.data.l[0] == STATE_TOGGLE) {
toggleState(*win, ce.data.l[1]);
toggleState(*win, ce.data.l[2]);
}
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
if (win == 0)
return true;
// ce.window = window to focus
2002-10-16 19:03:57 +00:00
2002-12-01 13:42:15 +00:00
// should move set focus somewhere else
// so we don't need fluxbox depedencies here
Fluxbox::instance()->setFocusedWindow(win);
return true;
} else if (ce.message_type == m_net_close_window) {
if (win == 0)
return true;
// ce.window = window to close (which in this case is the win argument)
win->close();
return true;
} else if (ce.message_type == m_net_moveresize_window) {
if (win == 0)
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
win->moveResize(ce.data.l[1], ce.data.l[2],
2002-12-01 13:42:15 +00:00
ce.data.l[3], ce.data.l[4]);
return true;
}
// we didn't handle the ce.message_type here
return false;
2002-10-02 16:26:05 +00:00
}
void Ewmh::createAtoms() {
2002-12-01 13:42:15 +00:00
Display *disp = BaseDisplay::getXDisplay();
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-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);
m_net_wm_window_type = XInternAtom(disp, "_NET_WM_WINDOW_TYPE", False);
2002-10-11 10:20:33 +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);
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
}
2002-10-11 10:20:33 +00:00
// set window state
void Ewmh::setState(FluxboxWindow &win, Atom state, bool value) const {
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();
}
2002-10-11 10:20:33 +00:00
}
// toggle window state
void Ewmh::toggleState(FluxboxWindow &win, Atom state) const {
2002-12-01 13:42:15 +00:00
if (state == m_net_wm_state_sticky) {
win.stick();
} else if (state == m_net_wm_state_shaded)
win.shade();
2002-10-11 10:20:33 +00:00
}