2003-02-01 10:03:29 +00:00
|
|
|
##############################################################################
|
|
|
|
### The history window placement algorithm. ebind historyplacement.place ###
|
|
|
|
### to the ob.EventAction.PlaceWindow event to use it. ###
|
|
|
|
##############################################################################
|
|
|
|
|
2003-02-19 19:28:11 +00:00
|
|
|
import windowplacement, config
|
2003-02-04 11:03:57 +00:00
|
|
|
|
|
|
|
def place(data):
|
|
|
|
"""Place a window usingthe history placement algorithm."""
|
|
|
|
_place(data)
|
|
|
|
|
2003-02-19 19:28:11 +00:00
|
|
|
export_functions = place
|
|
|
|
|
|
|
|
##############################################################################
|
|
|
|
|
|
|
|
config.add('historyplacement',
|
|
|
|
'ignore_requested_positions',
|
|
|
|
'Ignore Requested Positions',
|
|
|
|
"When true, the placement algorithm will attempt to place " + \
|
|
|
|
"windows even when they request a position (like XMMS can)." + \
|
|
|
|
"Note this only applies to 'normal' windows, not to special " + \
|
|
|
|
"cases like desktops and docks.",
|
|
|
|
'boolean',
|
|
|
|
0)
|
|
|
|
config.add('historyplacement',
|
|
|
|
'dont_duplicate',
|
|
|
|
"Don't Diplicate",
|
|
|
|
"When true, if 2 copies of the same match in history are to be " + \
|
|
|
|
"placed before one of them is closed (so it would be placed " + \
|
|
|
|
"over-top of the last one), this will cause the second window to "+\
|
|
|
|
"not be placed via history, and the 'Fallback Algorithm' will be "+\
|
|
|
|
"used instead.",
|
|
|
|
'boolean',
|
|
|
|
1)
|
|
|
|
config.add('historyplacement',
|
|
|
|
'filename',
|
|
|
|
'History Database Filename',
|
|
|
|
"The name of the file where history data will be stored. The " + \
|
|
|
|
"number of the screen is appended onto this name. The file will " +\
|
|
|
|
"be placed in ~/.openbox/.",
|
|
|
|
'string',
|
|
|
|
'historydb')
|
|
|
|
config.add('historyplacement',
|
|
|
|
'fallback',
|
|
|
|
'Fallback Algorithm',
|
|
|
|
"The window placement algorithm that will be used when history " + \
|
|
|
|
"placement does not have a place for the window.",
|
|
|
|
'enum',
|
|
|
|
windowplacement.random,
|
|
|
|
options = windowplacement.export_functions)
|
|
|
|
config.add('historyplacement',
|
|
|
|
'confirm_callback',
|
|
|
|
'Confirm Placement Callback',
|
|
|
|
"A function which will be called before attempting to place a " + \
|
|
|
|
"window via history. If the function returns true, then an " + \
|
|
|
|
"attempt will be made to place the window. If it returns false, " +\
|
|
|
|
"the 'Fallback Algorithm' will be directly applied instead. The " +\
|
|
|
|
"function must take 1 argument, which will be the callback data " +\
|
|
|
|
"which was passed to invoke the window placement.",
|
|
|
|
'function',
|
|
|
|
None)
|
|
|
|
|
2003-02-04 11:03:57 +00:00
|
|
|
###########################################################################
|
|
|
|
|
|
|
|
###########################################################################
|
|
|
|
### Internal stuff, should not be accessed outside the module. ###
|
|
|
|
###########################################################################
|
2003-02-01 10:03:29 +00:00
|
|
|
|
|
|
|
import otk
|
|
|
|
import ob
|
|
|
|
import os
|
|
|
|
import string
|
|
|
|
|
|
|
|
_data = []
|
|
|
|
|
|
|
|
class _state:
|
|
|
|
def __init__(self, appname, appclass, role, x, y):
|
|
|
|
self.appname = appname
|
|
|
|
self.appclass = appclass
|
|
|
|
self.role = role
|
|
|
|
self.x = x
|
|
|
|
self.y = y
|
2003-02-10 04:01:04 +00:00
|
|
|
self.placed = 0
|
2003-02-01 10:03:29 +00:00
|
|
|
def __eq__(self, other):
|
|
|
|
if self.appname == other.appname and \
|
|
|
|
self.appclass == other.appclass and \
|
|
|
|
self.role == other.role:
|
|
|
|
return 1
|
|
|
|
return 0
|
|
|
|
|
|
|
|
def _load(data):
|
2003-02-01 10:36:34 +00:00
|
|
|
global _data
|
2003-02-06 15:17:50 +00:00
|
|
|
try:
|
2003-02-19 19:28:11 +00:00
|
|
|
file = open(os.environ['HOME'] + '/.openbox/' + \
|
|
|
|
config.get('historyplacement', 'filename') + \
|
|
|
|
"." + str(data.screen), 'r')
|
2003-02-01 10:03:29 +00:00
|
|
|
# read data
|
|
|
|
for line in file.readlines():
|
|
|
|
line = line[:-1] # drop the '\n'
|
|
|
|
try:
|
|
|
|
s = string.split(line, '\0')
|
|
|
|
state = _state(s[0], s[1], s[2],
|
|
|
|
string.atoi(s[3]), string.atoi(s[4]))
|
|
|
|
|
|
|
|
while len(_data)-1 < data.screen:
|
|
|
|
_data.append([])
|
|
|
|
_data[data.screen].append(state)
|
|
|
|
|
2003-02-04 12:44:03 +00:00
|
|
|
except ValueError: pass
|
|
|
|
except IndexError: pass
|
2003-02-01 10:03:29 +00:00
|
|
|
file.close()
|
2003-02-06 15:17:50 +00:00
|
|
|
except IOError: pass
|
2003-02-01 10:03:29 +00:00
|
|
|
|
|
|
|
def _save(data):
|
2003-02-01 10:36:34 +00:00
|
|
|
global _data
|
2003-02-19 19:28:11 +00:00
|
|
|
file = open(os.environ['HOME']+'/.openbox/'+ \
|
|
|
|
config.get('historyplacement', 'filename') + \
|
|
|
|
"." + str(data.screen), 'w')
|
2003-02-01 10:03:29 +00:00
|
|
|
if file:
|
|
|
|
while len(_data)-1 < data.screen:
|
|
|
|
_data.append([])
|
|
|
|
for i in _data[data.screen]:
|
|
|
|
file.write(i.appname + '\0' +
|
|
|
|
i.appclass + '\0' +
|
|
|
|
i.role + '\0' +
|
|
|
|
str(i.x) + '\0' +
|
|
|
|
str(i.y) + '\n')
|
|
|
|
file.close()
|
|
|
|
|
2003-02-01 10:36:34 +00:00
|
|
|
def _create_state(data):
|
|
|
|
global _data
|
|
|
|
area = data.client.area()
|
|
|
|
return _state(data.client.appName(), data.client.appClass(),
|
|
|
|
data.client.role(), area.x(), area.y())
|
|
|
|
|
|
|
|
def _find(screen, state):
|
|
|
|
global _data
|
|
|
|
try:
|
|
|
|
return _data[screen].index(state)
|
|
|
|
except ValueError:
|
|
|
|
return -1
|
|
|
|
except IndexError:
|
|
|
|
while len(_data)-1 < screen:
|
|
|
|
_data.append([])
|
|
|
|
return _find(screen, state) # try again
|
|
|
|
|
2003-02-04 11:03:57 +00:00
|
|
|
def _place(data):
|
2003-02-01 10:36:34 +00:00
|
|
|
global _data
|
2003-02-01 10:03:29 +00:00
|
|
|
if data.client:
|
2003-02-19 19:28:11 +00:00
|
|
|
if not (config.get('historyplacement', 'ignore_requested_positions') \
|
|
|
|
and data.client.normal()):
|
2003-02-01 10:36:34 +00:00
|
|
|
if data.client.positionRequested(): return
|
|
|
|
state = _create_state(data)
|
2003-02-01 11:04:46 +00:00
|
|
|
try:
|
2003-02-19 19:28:11 +00:00
|
|
|
confirm = config.get('historyplacement', 'confirm_callback')
|
|
|
|
if not confirm or confirm(data):
|
2003-02-01 11:04:46 +00:00
|
|
|
print "looking for : " + state.appname + " : " + \
|
|
|
|
state.appclass + " : " + state.role
|
|
|
|
|
|
|
|
i = _find(data.screen, state)
|
|
|
|
if i >= 0:
|
|
|
|
coords = _data[data.screen][i]
|
|
|
|
print "Found in history ("+str(coords.x)+","+\
|
|
|
|
str(coords.y)+")"
|
2003-02-19 19:28:11 +00:00
|
|
|
if not (config.get('historyplacement', 'dont_duplicate') \
|
|
|
|
and coords.placed):
|
2003-02-10 04:01:04 +00:00
|
|
|
data.client.move(coords.x, coords.y)
|
|
|
|
coords.placed = 1
|
|
|
|
return
|
|
|
|
else:
|
|
|
|
print "Already placed another window there"
|
2003-02-01 11:04:46 +00:00
|
|
|
else:
|
|
|
|
print "No match in history"
|
|
|
|
except TypeError:
|
|
|
|
pass
|
2003-02-19 19:28:11 +00:00
|
|
|
fallback = config.get('historyplacement', 'fallback')
|
|
|
|
if fallback: fallback(data)
|
2003-02-01 10:03:29 +00:00
|
|
|
|
|
|
|
def _save_window(data):
|
2003-02-01 10:36:34 +00:00
|
|
|
global _data
|
2003-02-01 10:03:29 +00:00
|
|
|
if data.client:
|
2003-02-01 10:36:34 +00:00
|
|
|
state = _create_state(data)
|
|
|
|
print "looking for : " + state.appname + " : " + state.appclass + \
|
|
|
|
" : " + state.role
|
|
|
|
|
|
|
|
i = _find(data.screen, state)
|
|
|
|
if i >= 0:
|
2003-02-01 10:03:29 +00:00
|
|
|
print "replacing"
|
|
|
|
_data[data.screen][i] = state # replace it
|
2003-02-01 10:36:34 +00:00
|
|
|
else:
|
2003-02-01 10:03:29 +00:00
|
|
|
print "appending"
|
|
|
|
_data[data.screen].append(state)
|
|
|
|
|
|
|
|
ob.ebind(ob.EventAction.CloseWindow, _save_window)
|
|
|
|
ob.ebind(ob.EventAction.Startup, _load)
|
|
|
|
ob.ebind(ob.EventAction.Shutdown, _save)
|
|
|
|
|
|
|
|
print "Loaded historyplacement.py"
|