2003-01-03 06:43:53 +00:00
|
|
|
###########################################################################
|
|
|
|
### Functions that can be used as callbacks for mouse/keyboard bindings ###
|
|
|
|
###########################################################################
|
|
|
|
|
2003-01-08 17:03:36 +00:00
|
|
|
def state_above(data, add=2):
|
|
|
|
"""Toggles, adds or removes the 'above' state on a window."""
|
2003-01-10 03:11:48 +00:00
|
|
|
if not data.client: return
|
2003-01-13 10:09:06 +00:00
|
|
|
send_client_msg(display.screenInfo(data.screen).rootWindow(),
|
2003-01-16 08:44:52 +00:00
|
|
|
Property_atoms().net_wm_state, data.client.window(), add,
|
|
|
|
Property_atoms().net_wm_state_above)
|
2003-01-08 17:03:36 +00:00
|
|
|
|
|
|
|
def state_below(data, add=2):
|
|
|
|
"""Toggles, adds or removes the 'below' state on a window."""
|
2003-01-10 03:11:48 +00:00
|
|
|
if not data.client: return
|
2003-01-13 10:09:06 +00:00
|
|
|
send_client_msg(display.screenInfo(data.screen).rootWindow(),
|
2003-01-16 08:44:52 +00:00
|
|
|
Property_atoms().net_wm_state, data.client.window(), add,
|
|
|
|
Property_atoms().net_wm_state_below)
|
2003-01-08 17:03:36 +00:00
|
|
|
|
|
|
|
def state_shaded(data, add=2):
|
|
|
|
"""Toggles, adds or removes the 'shaded' state on a window."""
|
2003-01-10 03:11:48 +00:00
|
|
|
if not data.client: return
|
2003-01-13 10:09:06 +00:00
|
|
|
send_client_msg(display.screenInfo(data.screen).rootWindow(),
|
2003-01-16 08:44:52 +00:00
|
|
|
Property_atoms().net_wm_state, data.client.window(), add,
|
|
|
|
Property_atoms().net_wm_state_shaded)
|
2003-01-08 17:03:36 +00:00
|
|
|
|
2003-01-03 06:43:53 +00:00
|
|
|
def close(data):
|
|
|
|
"""Closes the window on which the event occured"""
|
2003-01-10 03:11:48 +00:00
|
|
|
if not data.client: return
|
2003-01-13 10:09:06 +00:00
|
|
|
send_client_msg(display.screenInfo(data.screen).rootWindow(),
|
2003-01-16 08:44:52 +00:00
|
|
|
Property_atoms().net_close_window, data.client.window(), 0)
|
2003-01-03 06:43:53 +00:00
|
|
|
|
|
|
|
def focus(data):
|
|
|
|
"""Focuses the window on which the event occured"""
|
2003-01-10 03:11:48 +00:00
|
|
|
if not data.client: return
|
2003-01-04 08:41:42 +00:00
|
|
|
# !normal windows dont get focus from window enter events
|
2003-01-10 03:11:48 +00:00
|
|
|
if data.action == EventEnterWindow and not data.client.normal():
|
2003-01-04 08:41:42 +00:00
|
|
|
return
|
2003-01-10 03:11:48 +00:00
|
|
|
data.client.focus()
|
2003-01-03 06:43:53 +00:00
|
|
|
|
|
|
|
def move(data):
|
|
|
|
"""Moves the window interactively. This should only be used with
|
|
|
|
MouseMotion events"""
|
2003-01-10 03:11:48 +00:00
|
|
|
if not data.client: return
|
2003-01-03 06:43:53 +00:00
|
|
|
|
2003-01-04 08:41:42 +00:00
|
|
|
# !normal windows dont get moved
|
2003-01-10 03:11:48 +00:00
|
|
|
if not data.client.normal(): return
|
2003-01-04 08:06:48 +00:00
|
|
|
|
2003-01-10 03:11:48 +00:00
|
|
|
dx = data.xroot - data.pressx
|
|
|
|
dy = data.yroot - data.pressy
|
|
|
|
data.client.move(data.press_clientx + dx, data.press_clienty + dy)
|
2003-01-03 06:43:53 +00:00
|
|
|
|
|
|
|
def resize(data):
|
|
|
|
"""Resizes the window interactively. This should only be used with
|
|
|
|
MouseMotion events"""
|
2003-01-10 03:11:48 +00:00
|
|
|
if not data.client: return
|
2003-01-03 06:43:53 +00:00
|
|
|
|
2003-01-04 08:41:42 +00:00
|
|
|
# !normal windows dont get moved
|
2003-01-10 03:11:48 +00:00
|
|
|
if not data.client.normal(): return
|
2003-01-04 08:06:48 +00:00
|
|
|
|
2003-01-10 03:11:48 +00:00
|
|
|
px = data.pressx
|
|
|
|
py = data.pressy
|
|
|
|
dx = data.xroot - px
|
|
|
|
dy = data.yroot - py
|
2003-01-03 06:43:53 +00:00
|
|
|
|
|
|
|
# pick a corner to anchor
|
2003-01-10 03:11:48 +00:00
|
|
|
if not (resize_nearest or data.context == MC_Grip):
|
2003-01-11 23:16:27 +00:00
|
|
|
corner = Client.TopLeft
|
2003-01-03 06:43:53 +00:00
|
|
|
else:
|
2003-01-10 03:11:48 +00:00
|
|
|
x = px - data.press_clientx
|
|
|
|
y = py - data.press_clienty
|
|
|
|
if y < data.press_clientheight / 2:
|
|
|
|
if x < data.press_clientwidth / 2:
|
2003-01-11 23:16:27 +00:00
|
|
|
corner = Client.BottomRight
|
2003-01-03 06:43:53 +00:00
|
|
|
dx *= -1
|
|
|
|
else:
|
2003-01-11 23:16:27 +00:00
|
|
|
corner = Client.BottomLeft
|
2003-01-03 06:43:53 +00:00
|
|
|
dy *= -1
|
|
|
|
else:
|
2003-01-10 03:11:48 +00:00
|
|
|
if x < data.press_clientwidth / 2:
|
2003-01-11 23:16:27 +00:00
|
|
|
corner = Client.TopRight
|
2003-01-03 06:43:53 +00:00
|
|
|
dx *= -1
|
|
|
|
else:
|
2003-01-11 23:16:27 +00:00
|
|
|
corner = Client.TopLeft
|
2003-01-03 06:43:53 +00:00
|
|
|
|
2003-01-10 03:11:48 +00:00
|
|
|
data.client.resize(corner,
|
|
|
|
data.press_clientwidth + dx,
|
|
|
|
data.press_clientheight + dy);
|
2003-01-03 06:43:53 +00:00
|
|
|
|
2003-01-10 22:50:56 +00:00
|
|
|
def restart(data, other = ""):
|
|
|
|
"""Restarts openbox, optionally starting another window manager."""
|
|
|
|
openbox.restart(other)
|
2003-01-03 22:06:08 +00:00
|
|
|
|
2003-01-03 18:21:28 +00:00
|
|
|
def raise_win(data):
|
2003-01-04 18:48:09 +00:00
|
|
|
"""Raises the window on which the event occured"""
|
2003-01-10 03:11:48 +00:00
|
|
|
if not data.client: return
|
|
|
|
openbox.screen(data.screen).restack(1, data.client)
|
2003-01-03 18:21:28 +00:00
|
|
|
|
|
|
|
def lower_win(data):
|
2003-01-04 18:48:09 +00:00
|
|
|
"""Lowers the window on which the event occured"""
|
2003-01-10 03:11:48 +00:00
|
|
|
if not data.client: return
|
|
|
|
openbox.screen(data.screen).restack(0, data.client)
|
2003-01-03 22:25:52 +00:00
|
|
|
|
2003-01-03 23:51:06 +00:00
|
|
|
def toggle_shade(data):
|
2003-01-04 18:48:09 +00:00
|
|
|
"""Toggles the shade status of the window on which the event occured"""
|
2003-01-08 17:03:36 +00:00
|
|
|
state_shaded(data)
|
2003-01-03 23:51:06 +00:00
|
|
|
|
|
|
|
def shade(data):
|
2003-01-04 18:48:09 +00:00
|
|
|
"""Shades the window on which the event occured"""
|
2003-01-08 17:03:36 +00:00
|
|
|
state_shaded(data, 1)
|
2003-01-03 23:51:06 +00:00
|
|
|
|
|
|
|
def unshade(data):
|
2003-01-04 18:48:09 +00:00
|
|
|
"""Unshades the window on which the event occured"""
|
2003-01-08 17:03:36 +00:00
|
|
|
state_shaded(data, 0)
|
2003-01-07 02:24:43 +00:00
|
|
|
|
2003-01-07 04:51:55 +00:00
|
|
|
def change_desktop(data, num):
|
|
|
|
"""Switches to a specified desktop"""
|
2003-01-13 10:09:06 +00:00
|
|
|
root = display.screenInfo(data.screen).rootWindow()
|
2003-01-16 08:44:52 +00:00
|
|
|
send_client_msg(root, Property_atoms().net_current_desktop, root, num)
|
2003-01-07 04:51:55 +00:00
|
|
|
|
2003-01-07 02:27:51 +00:00
|
|
|
def next_desktop(data, no_wrap=0):
|
2003-01-07 02:46:14 +00:00
|
|
|
"""Switches to the next desktop, optionally (by default) cycling around to
|
|
|
|
the first when going past the last."""
|
2003-01-10 03:11:48 +00:00
|
|
|
screen = openbox.screen(data.screen)
|
|
|
|
d = screen.desktop()
|
|
|
|
n = screen.numDesktops()
|
2003-01-07 02:27:51 +00:00
|
|
|
if (d < (n-1)):
|
2003-01-07 02:24:43 +00:00
|
|
|
d = d + 1
|
2003-01-07 02:27:51 +00:00
|
|
|
elif not no_wrap:
|
|
|
|
d = 0
|
2003-01-07 04:51:55 +00:00
|
|
|
change_desktop(data, d)
|
2003-01-07 02:24:43 +00:00
|
|
|
|
2003-01-07 02:27:51 +00:00
|
|
|
def prev_desktop(data, no_wrap=0):
|
2003-01-07 02:46:14 +00:00
|
|
|
"""Switches to the previous desktop, optionally (by default) cycling around
|
|
|
|
to the last when going past the first."""
|
2003-01-10 03:11:48 +00:00
|
|
|
screen = openbox.screen(data.screen)
|
|
|
|
d = screen.desktop()
|
|
|
|
n = screen.numDesktops()
|
2003-01-07 02:24:43 +00:00
|
|
|
if (d > 0):
|
|
|
|
d = d - 1
|
2003-01-07 02:27:51 +00:00
|
|
|
elif not no_wrap:
|
2003-01-07 02:24:43 +00:00
|
|
|
d = n - 1
|
2003-01-07 04:51:55 +00:00
|
|
|
change_desktop(data, d)
|
2003-01-07 02:44:54 +00:00
|
|
|
|
2003-01-11 23:16:27 +00:00
|
|
|
def send_to_all_desktops(data):
|
|
|
|
"""Sends a client to all desktops"""
|
|
|
|
if not data.client: return
|
2003-01-13 10:09:06 +00:00
|
|
|
send_client_msg(display.screenInfo(data.screen).rootWindow(),
|
2003-01-16 08:44:52 +00:00
|
|
|
Property_atoms().net_wm_desktop, data.client.window(),
|
|
|
|
0xffffffff)
|
2003-01-11 23:16:27 +00:00
|
|
|
|
2003-01-07 04:51:55 +00:00
|
|
|
def send_to_desktop(data, num):
|
|
|
|
"""Sends a client to a specified desktop"""
|
2003-01-10 03:11:48 +00:00
|
|
|
if not data.client: return
|
2003-01-13 10:09:06 +00:00
|
|
|
send_client_msg(display.screenInfo(data.screen).rootWindow(),
|
2003-01-16 08:44:52 +00:00
|
|
|
Property_atoms().net_wm_desktop, data.client.window(), num)
|
2003-01-07 04:51:55 +00:00
|
|
|
|
|
|
|
def send_to_next_desktop(data, no_wrap=0, follow=1):
|
|
|
|
"""Sends a window to the next desktop, optionally (by default) cycling
|
|
|
|
around to the first when going past the last. Also optionally moving to
|
|
|
|
the new desktop after sending the window."""
|
2003-01-10 03:11:48 +00:00
|
|
|
if not data.client: return
|
|
|
|
screen = openbox.screen(data.screen)
|
|
|
|
d = screen.desktop()
|
|
|
|
n = screen.numDesktops()
|
2003-01-07 04:51:55 +00:00
|
|
|
if (d < (n-1)):
|
|
|
|
d = d + 1
|
|
|
|
elif not no_wrap:
|
|
|
|
d = 0
|
|
|
|
send_to_desktop(data, d)
|
|
|
|
if follow:
|
|
|
|
change_desktop(data, d)
|
|
|
|
|
|
|
|
def send_to_prev_desktop(data, no_wrap=0, follow=1):
|
|
|
|
"""Sends a window to the previous desktop, optionally (by default) cycling
|
|
|
|
around to the last when going past the first. Also optionally moving to
|
|
|
|
the new desktop after sending the window."""
|
2003-01-10 03:11:48 +00:00
|
|
|
if not data.client: return
|
|
|
|
screen = openbox.screen(data.screen)
|
|
|
|
d = screen.desktop()
|
|
|
|
n = screen.numDesktops()
|
2003-01-07 04:51:55 +00:00
|
|
|
if (d > 0):
|
|
|
|
d = d - 1
|
|
|
|
elif not no_wrap:
|
|
|
|
d = n - 1
|
|
|
|
send_to_desktop(data, d)
|
|
|
|
if follow:
|
|
|
|
change_desktop(data, d)
|
2003-01-08 16:29:39 +00:00
|
|
|
|
2003-01-04 02:34:58 +00:00
|
|
|
#########################################
|
|
|
|
### Convenience functions for scripts ###
|
|
|
|
#########################################
|
|
|
|
|
|
|
|
def execute(bin, screen = 0):
|
2003-01-04 18:48:09 +00:00
|
|
|
"""Executes a command on the specified screen. It is recommended that you
|
|
|
|
use this call instead of a python system call. If the specified screen
|
|
|
|
is beyond your range of screens, the default is used instead."""
|
2003-01-10 03:11:48 +00:00
|
|
|
openbox.execute(screen, bin)
|
2003-01-03 23:51:06 +00:00
|
|
|
|
2003-01-10 09:32:58 +00:00
|
|
|
def setup_click_focus(click_raise = 1):
|
|
|
|
"""Sets up for focusing windows by clicking on or in the window.
|
|
|
|
Optionally, clicking on or in a window can raise the window to the
|
|
|
|
front of its stacking layer."""
|
2003-01-11 20:50:47 +00:00
|
|
|
mbind("Left", MC_Titlebar, MousePress, focus)
|
|
|
|
mbind("Left", MC_Handle, MousePress, focus)
|
|
|
|
mbind("Left", MC_Grip, MousePress, focus)
|
|
|
|
mbind("Left", MC_Window, MousePress, focus)
|
2003-01-10 09:32:58 +00:00
|
|
|
if click_raise:
|
2003-01-11 20:50:47 +00:00
|
|
|
mbind("Left", MC_Titlebar, MousePress, raise_win)
|
|
|
|
mbind("Left", MC_Handle, MousePress, raise_win)
|
|
|
|
mbind("Left", MC_Grip, MousePress, raise_win)
|
|
|
|
mbind("Left", MC_Window, MousePress, raise_win)
|
2003-01-10 09:32:58 +00:00
|
|
|
|
|
|
|
def setup_sloppy_focus(click_focus = 1, click_raise = 0):
|
|
|
|
"""Sets up for focusing windows when the mouse pointer enters them.
|
|
|
|
Optionally, clicking on or in a window can focus it if your pointer
|
|
|
|
ends up inside a window without focus. Also, optionally, clicking on or
|
|
|
|
in a window can raise the window to the front of its stacking layer."""
|
|
|
|
ebind(EventEnterWindow, focus)
|
|
|
|
if click_focus:
|
|
|
|
setup_click_focus(click_raise)
|
|
|
|
|
|
|
|
def setup_window_clicks():
|
|
|
|
"""Sets up the default bindings for various mouse buttons for various
|
|
|
|
contexts.
|
|
|
|
This includes:
|
|
|
|
* Alt-left drag anywhere on a window will move it
|
|
|
|
* Alt-right drag anywhere on a window will resize it
|
|
|
|
* Left drag on a window's titlebar/handle will move it
|
|
|
|
* Left drag on a window's handle grips will resize it
|
|
|
|
* Alt-left press anywhere on a window's will raise it to the front of
|
|
|
|
its stacking layer.
|
|
|
|
* Left press on a window's titlebar/handle will raise it to the front
|
|
|
|
of its stacking layer.
|
|
|
|
* Alt-middle click anywhere on a window's will lower it to the bottom
|
|
|
|
of its stacking layer.
|
|
|
|
* Middle click on a window's titlebar/handle will lower it to the
|
|
|
|
bottom of its stacking layer.
|
|
|
|
* Double-left click on a window's titlebar will toggle shading it
|
|
|
|
"""
|
2003-01-11 20:50:47 +00:00
|
|
|
mbind("A-Left", MC_Frame, MouseMotion, move)
|
|
|
|
mbind("Left", MC_Titlebar, MouseMotion, move)
|
|
|
|
mbind("Left", MC_Handle, MouseMotion, move)
|
2003-01-10 09:32:58 +00:00
|
|
|
|
2003-01-11 20:50:47 +00:00
|
|
|
mbind("A-Right", MC_Frame, MouseMotion, resize)
|
|
|
|
mbind("Left", MC_Grip, MouseMotion, resize)
|
2003-01-10 09:32:58 +00:00
|
|
|
|
2003-01-11 20:50:47 +00:00
|
|
|
mbind("Left", MC_Titlebar, MousePress, raise_win)
|
|
|
|
mbind("Left", MC_Handle, MousePress, raise_win)
|
|
|
|
mbind("A-Left", MC_Frame, MousePress, raise_win)
|
|
|
|
mbind("A-Middle", MC_Frame, MouseClick, lower_win)
|
2003-01-12 00:14:23 +00:00
|
|
|
mbind("Middle", MC_Titlebar, MouseClick, lower_win)
|
|
|
|
mbind("Middle", MC_Handle, MouseClick, lower_win)
|
2003-01-10 09:32:58 +00:00
|
|
|
|
2003-01-11 20:50:47 +00:00
|
|
|
mbind("Left", MC_Titlebar, MouseDoubleClick, toggle_shade)
|
2003-01-10 09:32:58 +00:00
|
|
|
|
|
|
|
def setup_window_buttons():
|
|
|
|
"""Sets up the default behaviors for the buttons in the window titlebar."""
|
2003-01-11 20:50:47 +00:00
|
|
|
mbind("Left", MC_CloseButton, MouseClick, close)
|
2003-01-10 09:32:58 +00:00
|
|
|
|
|
|
|
def setup_scroll():
|
|
|
|
"""Sets up the default behaviors for the mouse scroll wheel.
|
|
|
|
This includes:
|
|
|
|
* scrolling on a window titlebar will shade/unshade it
|
|
|
|
* alt-scrolling anywhere will switch to the next/previous desktop
|
|
|
|
* control-alt-scrolling on a window will send it to the next/previous
|
|
|
|
desktop, and switch to the desktop with the window
|
|
|
|
"""
|
2003-01-11 20:50:47 +00:00
|
|
|
mbind("Up", MC_Titlebar, MouseClick, shade)
|
|
|
|
mbind("Down", MC_Titlebar, MouseClick, unshade)
|
2003-01-10 09:32:58 +00:00
|
|
|
|
2003-01-11 20:50:47 +00:00
|
|
|
mbind("A-Up", MC_Frame, MouseClick, next_desktop)
|
|
|
|
mbind("A-Up", MC_Root, MouseClick, next_desktop)
|
|
|
|
mbind("A-Down", MC_Frame, MouseClick, prev_desktop)
|
|
|
|
mbind("A-Down", MC_Root, MouseClick, prev_desktop)
|
2003-01-10 09:32:58 +00:00
|
|
|
|
2003-01-11 20:50:47 +00:00
|
|
|
mbind("C-A-Up", MC_Frame, MouseClick, send_to_next_desktop)
|
|
|
|
mbind("C-A-Down", MC_Frame, MouseClick, send_to_prev_desktop)
|
2003-01-10 09:32:58 +00:00
|
|
|
|
2003-01-10 09:44:59 +00:00
|
|
|
def setup_fallback_focus():
|
|
|
|
"""Sets up a focus fallback routine so that when no windows are focused,
|
2003-01-10 09:55:44 +00:00
|
|
|
the last window to have focus on the desktop will be focused."""
|
2003-01-10 09:47:43 +00:00
|
|
|
focus_stack = []
|
2003-01-10 09:44:59 +00:00
|
|
|
def focused(data):
|
2003-01-10 09:47:43 +00:00
|
|
|
#global focus_stack
|
2003-01-10 09:44:59 +00:00
|
|
|
if data.client:
|
|
|
|
window = data.client.window()
|
|
|
|
# add to front the stack
|
|
|
|
if window in focus_stack:
|
|
|
|
focus_stack.remove(window)
|
|
|
|
focus_stack.insert(0, window)
|
|
|
|
else:
|
|
|
|
# pass around focus
|
|
|
|
desktop = openbox.screen(data.screen).desktop()
|
|
|
|
l = len(focus_stack)
|
|
|
|
i = 0
|
|
|
|
while i < l:
|
|
|
|
w = focus_stack[i]
|
|
|
|
client = openbox.findClient(w)
|
|
|
|
if not client: # window is gone, remove it
|
|
|
|
focus_stack.pop(i)
|
|
|
|
l = l - 1
|
|
|
|
elif client.desktop() == desktop and \
|
|
|
|
client.normal() and client.focus():
|
|
|
|
break
|
|
|
|
else:
|
|
|
|
i = i + 1
|
|
|
|
|
|
|
|
ebind(EventFocus, focused)
|
|
|
|
|
|
|
|
|
2003-01-10 09:38:51 +00:00
|
|
|
############################################################################
|
|
|
|
### Window placement algorithms, choose one of these and ebind it to the ###
|
|
|
|
### EventPlaceWindow action. ###
|
|
|
|
############################################################################
|
|
|
|
|
|
|
|
ob_rand = None
|
|
|
|
import random
|
|
|
|
def placewindows_random(data):
|
|
|
|
if not data.client: return
|
|
|
|
client_area = data.client.area()
|
2003-01-13 10:09:06 +00:00
|
|
|
screen = display.screenInfo(data.screen)
|
2003-01-10 09:38:51 +00:00
|
|
|
width = screen.width() - client_area.width()
|
|
|
|
height = screen.height() - client_area.height()
|
|
|
|
global ob_rand
|
|
|
|
if not ob_rand: ob_rand = random.Random()
|
|
|
|
x = ob_rand.randrange(0, width-1)
|
|
|
|
y = ob_rand.randrange(0, height-1)
|
|
|
|
data.client.move(x, y)
|
|
|
|
|
|
|
|
|
2003-01-03 22:25:52 +00:00
|
|
|
print "Loaded builtins.py"
|