add the --replace command line option, and support for the WM_Sn selection

This commit is contained in:
Dana Jansens 2003-07-22 18:09:41 +00:00
parent 6a7d4de789
commit 845833226b
4 changed files with 138 additions and 31 deletions

View file

@ -186,6 +186,9 @@ static Window event_get_window(XEvent *e)
/* pick a window */ /* pick a window */
switch (e->type) { switch (e->type) {
case SelectionClear:
window = RootWindow(ob_display, ob_screen);
break;
case MapRequest: case MapRequest:
window = e->xmap.window; window = e->xmap.window;
break; break;
@ -548,6 +551,11 @@ static void event_handle_root(XEvent *e)
Atom msgtype; Atom msgtype;
switch(e->type) { switch(e->type) {
case SelectionClear:
g_message("Another WM has requested to replace us. Exiting.");
ob_exit();
break;
case ClientMessage: case ClientMessage:
if (e->xclient.format != 32) break; if (e->xclient.format != 32) break;

View file

@ -59,14 +59,15 @@ RrInstance *ob_rr_inst;
RrTheme *ob_rr_theme; RrTheme *ob_rr_theme;
Display *ob_display; Display *ob_display;
gint ob_screen; gint ob_screen;
Cursor ob_cursors[OB_NUM_CURSORS]; gboolean ob_replace_wm;
KeyCode ob_keys[OB_NUM_KEYS];
static ObState state; static ObState state;
static gboolean xsync; static gboolean xsync;
static gboolean shutdown; static gboolean shutdown;
static gboolean restart; static gboolean restart;
static char *restart_path; static char *restart_path;
static Cursor cursors[OB_NUM_CURSORS];
static KeyCode keys[OB_NUM_KEYS];
static void signal_handler(const ObEvent *e, void *data); static void signal_handler(const ObEvent *e, void *data);
static void parse_args(int argc, char **argv); static void parse_args(int argc, char **argv);
@ -169,41 +170,41 @@ int main(int argc, char **argv)
putenv(g_strdup_printf("DISPLAY=%s", DisplayString(ob_display))); putenv(g_strdup_printf("DISPLAY=%s", DisplayString(ob_display)));
/* create available cursors */ /* create available cursors */
ob_cursors[OB_CURSOR_POINTER] = cursors[OB_CURSOR_POINTER] =
XCreateFontCursor(ob_display, XC_left_ptr); XCreateFontCursor(ob_display, XC_left_ptr);
ob_cursors[OB_CURSOR_BUSY] = cursors[OB_CURSOR_BUSY] =
XCreateFontCursor(ob_display, XC_watch); XCreateFontCursor(ob_display, XC_watch);
ob_cursors[OB_CURSOR_MOVE] = cursors[OB_CURSOR_MOVE] =
XCreateFontCursor(ob_display, XC_fleur); XCreateFontCursor(ob_display, XC_fleur);
ob_cursors[OB_CURSOR_NORTH] = cursors[OB_CURSOR_NORTH] =
XCreateFontCursor(ob_display, XC_top_side); XCreateFontCursor(ob_display, XC_top_side);
ob_cursors[OB_CURSOR_NORTHEAST] = cursors[OB_CURSOR_NORTHEAST] =
XCreateFontCursor(ob_display, XC_top_right_corner); XCreateFontCursor(ob_display, XC_top_right_corner);
ob_cursors[OB_CURSOR_EAST] = cursors[OB_CURSOR_EAST] =
XCreateFontCursor(ob_display, XC_right_side); XCreateFontCursor(ob_display, XC_right_side);
ob_cursors[OB_CURSOR_SOUTHEAST] = cursors[OB_CURSOR_SOUTHEAST] =
XCreateFontCursor(ob_display, XC_bottom_right_corner); XCreateFontCursor(ob_display, XC_bottom_right_corner);
ob_cursors[OB_CURSOR_SOUTH] = cursors[OB_CURSOR_SOUTH] =
XCreateFontCursor(ob_display, XC_bottom_side); XCreateFontCursor(ob_display, XC_bottom_side);
ob_cursors[OB_CURSOR_SOUTHWEST] = cursors[OB_CURSOR_SOUTHWEST] =
XCreateFontCursor(ob_display, XC_bottom_left_corner); XCreateFontCursor(ob_display, XC_bottom_left_corner);
ob_cursors[OB_CURSOR_WEST] = cursors[OB_CURSOR_WEST] =
XCreateFontCursor(ob_display, XC_left_side); XCreateFontCursor(ob_display, XC_left_side);
ob_cursors[OB_CURSOR_NORTHWEST] = cursors[OB_CURSOR_NORTHWEST] =
XCreateFontCursor(ob_display, XC_top_left_corner); XCreateFontCursor(ob_display, XC_top_left_corner);
/* create available keycodes */ /* create available keycodes */
ob_keys[OB_KEY_RETURN] = keys[OB_KEY_RETURN] =
XKeysymToKeycode(ob_display, XStringToKeysym("Return")); XKeysymToKeycode(ob_display, XStringToKeysym("Return"));
ob_keys[OB_KEY_ESCAPE] = keys[OB_KEY_ESCAPE] =
XKeysymToKeycode(ob_display, XStringToKeysym("Escape")); XKeysymToKeycode(ob_display, XStringToKeysym("Escape"));
ob_keys[OB_KEY_LEFT] = keys[OB_KEY_LEFT] =
XKeysymToKeycode(ob_display, XStringToKeysym("Left")); XKeysymToKeycode(ob_display, XStringToKeysym("Left"));
ob_keys[OB_KEY_RIGHT] = keys[OB_KEY_RIGHT] =
XKeysymToKeycode(ob_display, XStringToKeysym("Right")); XKeysymToKeycode(ob_display, XStringToKeysym("Right"));
ob_keys[OB_KEY_UP] = keys[OB_KEY_UP] =
XKeysymToKeycode(ob_display, XStringToKeysym("Up")); XKeysymToKeycode(ob_display, XStringToKeysym("Up"));
ob_keys[OB_KEY_DOWN] = keys[OB_KEY_DOWN] =
XKeysymToKeycode(ob_display, XStringToKeysym("Down")); XKeysymToKeycode(ob_display, XStringToKeysym("Down"));
prop_startup(); /* get atoms values for the display */ prop_startup(); /* get atoms values for the display */
@ -503,6 +504,8 @@ static void print_help()
g_print(" --sm-client-id ID Specify session management ID\n"); g_print(" --sm-client-id ID Specify session management ID\n");
g_print(" --sm-disable Disable connection to session manager\n"); g_print(" --sm-disable Disable connection to session manager\n");
#endif #endif
g_print(" --replace Replace the currently running window "
"manager\n");
g_print(" --help Display this help and exit\n"); g_print(" --help Display this help and exit\n");
g_print(" --version Display the version and exit\n"); g_print(" --version Display the version and exit\n");
g_print(" --sync Run in synchronous mode (this is slow and\n" g_print(" --sync Run in synchronous mode (this is slow and\n"
@ -523,6 +526,8 @@ static void parse_args(int argc, char **argv)
exit(0); exit(0);
} else if (!strcmp(argv[i], "--g-fatal-warnings")) { } else if (!strcmp(argv[i], "--g-fatal-warnings")) {
g_log_set_always_fatal(G_LOG_LEVEL_CRITICAL); g_log_set_always_fatal(G_LOG_LEVEL_CRITICAL);
} else if (!strcmp(argv[i], "--replace")) {
ob_replace_wm = TRUE;
} else if (!strcmp(argv[i], "--sync")) { } else if (!strcmp(argv[i], "--sync")) {
xsync = TRUE; xsync = TRUE;
#ifdef USE_SM #ifdef USE_SM
@ -594,13 +599,13 @@ void ob_exit()
Cursor ob_cursor(ObCursor cursor) Cursor ob_cursor(ObCursor cursor)
{ {
g_assert(cursor < OB_NUM_CURSORS); g_assert(cursor < OB_NUM_CURSORS);
return ob_cursors[cursor]; return cursors[cursor];
} }
KeyCode ob_keycode(ObKey key) KeyCode ob_keycode(ObKey key)
{ {
g_assert(key < OB_NUM_KEYS); g_assert(key < OB_NUM_KEYS);
return ob_keys[key]; return keys[key];
} }
ObState ob_state() ObState ob_state()

View file

@ -27,6 +27,8 @@ SnDisplay *ob_sn_display;
/*! The number of the screen on which we're running */ /*! The number of the screen on which we're running */
extern gint ob_screen; extern gint ob_screen;
extern gboolean ob_replace_wm;
/* The state of execution of the window manager */ /* The state of execution of the window manager */
ObState ob_state(); ObState ob_state();

View file

@ -52,6 +52,91 @@ static void sn_event_func(SnMonitorEvent *event, void *data);
static void set_root_cursor(); static void set_root_cursor();
static gboolean replace_wm()
{
char *wm_sn;
Atom wm_sn_atom;
Window current_wm_sn_owner;
Time timestamp;
wm_sn = g_strdup_printf("WM_S%d", ob_screen);
wm_sn_atom = XInternAtom(ob_display, wm_sn, FALSE);
current_wm_sn_owner = XGetSelectionOwner(ob_display, wm_sn_atom);
if (current_wm_sn_owner) {
if (!ob_replace_wm) {
g_message("A window manager is already running on screen %d",
ob_screen);
return FALSE;
}
xerror_set_ignore(TRUE);
xerror_occured = FALSE;
/* We want to find out when the current selection owner dies */
XSelectInput(ob_display, current_wm_sn_owner, StructureNotifyMask);
XSync(ob_display, FALSE);
xerror_set_ignore(FALSE);
if (xerror_occured)
current_wm_sn_owner = None;
}
{
/* Generate a timestamp */
XEvent event;
XSelectInput(ob_display, screen_support_win, PropertyChangeMask);
XChangeProperty(ob_display, screen_support_win,
prop_atoms.wm_class, prop_atoms.string,
8, PropModeAppend, NULL, 0);
XWindowEvent(ob_display, screen_support_win,
PropertyChangeMask, &event);
XSelectInput(ob_display, screen_support_win, NoEventMask);
timestamp = event.xproperty.time;
}
XSetSelectionOwner(ob_display, wm_sn_atom, screen_support_win,
timestamp);
if (XGetSelectionOwner(ob_display, wm_sn_atom) != screen_support_win) {
g_message("Could not acquire window manager selection on screen %d",
ob_screen);
return FALSE;
}
/* Wait for old window manager to go away */
if (current_wm_sn_owner) {
XEvent event;
gulong wait = 0;
const gulong timeout = G_USEC_PER_SEC * 15; /* wait for 15s max */
while (wait < timeout) {
if (XCheckWindowEvent(ob_display, current_wm_sn_owner,
StructureNotifyMask, &event) &&
event.type == DestroyNotify)
break;
g_usleep(G_USEC_PER_SEC / 10);
wait += G_USEC_PER_SEC / 10;
}
if (wait >= timeout) {
g_message("Timeout expired while waiting for the current WM to die "
"on screen %d", ob_screen);
return FALSE;
}
}
/* Send client message indicating that we are now the WM */
prop_message(RootWindow(ob_display, ob_screen), prop_atoms.manager,
timestamp, wm_sn_atom, 0, 0, SubstructureNotifyMask);
return TRUE;
}
gboolean screen_annex() gboolean screen_annex()
{ {
XSetWindowAttributes attrib; XSetWindowAttributes attrib;
@ -59,6 +144,21 @@ gboolean screen_annex()
gint i, num_support; gint i, num_support;
guint32 *supported; guint32 *supported;
/* create the netwm support window */
attrib.override_redirect = TRUE;
screen_support_win = XCreateWindow(ob_display,
RootWindow(ob_display, ob_screen),
-100, -100, 1, 1, 0,
CopyFromParent, InputOutput,
CopyFromParent,
CWOverrideRedirect, &attrib);
XMapRaised(ob_display, screen_support_win);
if (!replace_wm()) {
XDestroyWindow(ob_display, screen_support_win);
return FALSE;
}
xerror_set_ignore(TRUE); xerror_set_ignore(TRUE);
xerror_occured = FALSE; xerror_occured = FALSE;
XSelectInput(ob_display, RootWindow(ob_display, ob_screen), XSelectInput(ob_display, RootWindow(ob_display, ob_screen),
@ -67,6 +167,8 @@ gboolean screen_annex()
if (xerror_occured) { if (xerror_occured) {
g_message("A window manager is already running on screen %d", g_message("A window manager is already running on screen %d",
ob_screen); ob_screen);
XDestroyWindow(ob_display, screen_support_win);
return FALSE; return FALSE;
} }
@ -80,16 +182,6 @@ gboolean screen_annex()
PROP_SET32(RootWindow(ob_display, ob_screen), PROP_SET32(RootWindow(ob_display, ob_screen),
openbox_pid, cardinal, pid); openbox_pid, cardinal, pid);
/* create the netwm support window */
attrib.override_redirect = TRUE;
screen_support_win = XCreateWindow(ob_display,
RootWindow(ob_display, ob_screen),
-100, -100, 1, 1, 0,
CopyFromParent, InputOutput,
CopyFromParent,
CWOverrideRedirect, &attrib);
XMapRaised(ob_display, screen_support_win);
/* set supporting window */ /* set supporting window */
PROP_SET32(RootWindow(ob_display, ob_screen), PROP_SET32(RootWindow(ob_display, ob_screen),
net_supporting_wm_check, window, screen_support_win); net_supporting_wm_check, window, screen_support_win);