2009-11-07 01:40:43 +00:00
|
|
|
#!/usr/bin/python
|
|
|
|
# Terminator by Chris Jones <cmsj@tenshu.net>
|
|
|
|
# GPL v2 only
|
|
|
|
"""paned.py - a base Paned container class and the vertical/horizontal
|
|
|
|
variants"""
|
|
|
|
|
|
|
|
import gobject
|
|
|
|
import gtk
|
|
|
|
|
2009-11-20 22:53:06 +00:00
|
|
|
from util import dbg, err, get_top_window
|
2010-01-11 20:06:53 +00:00
|
|
|
from terminator import Terminator
|
2009-11-25 00:37:29 +00:00
|
|
|
from factory import Factory
|
2009-11-07 01:40:43 +00:00
|
|
|
from container import Container
|
|
|
|
|
|
|
|
# pylint: disable-msg=R0921
|
2009-11-14 17:30:03 +00:00
|
|
|
# pylint: disable-msg=E1101
|
2009-11-07 01:40:43 +00:00
|
|
|
class Paned(Container):
|
|
|
|
"""Base class for Paned Containers"""
|
|
|
|
|
|
|
|
def __init__(self):
|
|
|
|
"""Class initialiser"""
|
|
|
|
self.terminator = Terminator()
|
2009-11-14 22:58:22 +00:00
|
|
|
Container.__init__(self)
|
2009-11-08 23:06:26 +00:00
|
|
|
self.signals.append({'name': 'resize-term',
|
|
|
|
'flags': gobject.SIGNAL_RUN_LAST,
|
|
|
|
'return_type': gobject.TYPE_NONE,
|
|
|
|
'param_types': (gobject.TYPE_STRING,)})
|
2009-11-07 01:40:43 +00:00
|
|
|
|
|
|
|
|
2009-11-14 17:30:03 +00:00
|
|
|
# pylint: disable-msg=W0613
|
2009-11-08 01:06:31 +00:00
|
|
|
def set_initial_position(self, widget, event):
|
|
|
|
"""Set the initial position of the widget"""
|
|
|
|
if isinstance(self, gtk.VPaned):
|
|
|
|
position = self.allocation.height / 2
|
|
|
|
else:
|
|
|
|
position = self.allocation.width / 2
|
|
|
|
|
2009-11-14 18:55:07 +00:00
|
|
|
dbg("Paned::set_initial_position: Setting position to: %d" % position)
|
2009-11-08 01:06:31 +00:00
|
|
|
self.set_position(position)
|
2010-01-18 13:17:35 +00:00
|
|
|
self.cnxids.remove_signal(self, 'expose-event')
|
2009-11-08 01:06:31 +00:00
|
|
|
|
2009-11-14 17:30:03 +00:00
|
|
|
# pylint: disable-msg=W0613
|
2010-03-05 22:44:38 +00:00
|
|
|
def split_axis(self, widget, vertical=True, cwd=None, sibling=None,
|
2010-03-02 20:38:28 +00:00
|
|
|
widgetfirst=True):
|
2009-11-07 01:40:43 +00:00
|
|
|
"""Default axis splitter. This should be implemented by subclasses"""
|
2010-02-27 12:30:38 +00:00
|
|
|
order = None
|
2010-02-27 10:26:30 +00:00
|
|
|
|
2009-11-25 00:37:29 +00:00
|
|
|
maker = Factory()
|
|
|
|
|
2009-11-08 01:06:31 +00:00
|
|
|
self.remove(widget)
|
|
|
|
if vertical:
|
|
|
|
container = VPaned()
|
|
|
|
else:
|
|
|
|
container = HPaned()
|
|
|
|
|
2009-11-17 04:56:55 +00:00
|
|
|
if not sibling:
|
2009-11-25 00:37:29 +00:00
|
|
|
sibling = maker.make('terminal')
|
2010-03-05 22:44:38 +00:00
|
|
|
sibling.set_cwd(cwd)
|
2010-02-04 00:59:11 +00:00
|
|
|
sibling.spawn_child()
|
2009-11-08 01:06:31 +00:00
|
|
|
|
2009-11-14 18:55:07 +00:00
|
|
|
self.add(container)
|
|
|
|
self.show_all()
|
|
|
|
|
2010-02-27 12:30:38 +00:00
|
|
|
order = [widget, sibling]
|
2010-03-02 20:38:28 +00:00
|
|
|
if widgetfirst is False:
|
2010-02-27 12:30:38 +00:00
|
|
|
order.reverse()
|
2010-02-27 10:26:30 +00:00
|
|
|
|
2010-02-27 12:30:38 +00:00
|
|
|
for terminal in order:
|
|
|
|
container.add(terminal)
|
2009-11-08 01:06:31 +00:00
|
|
|
|
|
|
|
self.show_all()
|
2009-11-07 01:40:43 +00:00
|
|
|
|
|
|
|
def add(self, widget):
|
|
|
|
"""Add a widget to the container"""
|
2009-11-25 00:37:29 +00:00
|
|
|
maker = Factory()
|
2009-11-07 01:40:43 +00:00
|
|
|
if len(self.children) == 0:
|
|
|
|
self.pack1(widget, True, True)
|
|
|
|
self.children.append(widget)
|
|
|
|
elif len(self.children) == 1:
|
2009-11-08 01:06:31 +00:00
|
|
|
if self.get_child1():
|
|
|
|
self.pack2(widget, True, True)
|
|
|
|
else:
|
|
|
|
self.pack1(widget, True, True)
|
2009-11-07 01:40:43 +00:00
|
|
|
self.children.append(widget)
|
|
|
|
else:
|
2009-11-14 22:58:22 +00:00
|
|
|
raise ValueError('Paned widgets can only have two children')
|
2009-11-07 01:40:43 +00:00
|
|
|
|
2009-11-25 00:37:29 +00:00
|
|
|
if maker.isinstance(widget, 'Terminal'):
|
2009-11-20 22:53:06 +00:00
|
|
|
top_window = get_top_window(self)
|
2009-11-14 22:58:22 +00:00
|
|
|
signals = {'close-term': self.wrapcloseterm,
|
|
|
|
'split-horiz': self.split_horiz,
|
|
|
|
'split-vert': self.split_vert,
|
2009-12-10 13:20:03 +00:00
|
|
|
'title-change': self.propagate_title_change,
|
2009-11-14 22:58:22 +00:00
|
|
|
'resize-term': self.resizeterm,
|
2010-01-24 12:55:03 +00:00
|
|
|
'zoom': top_window.zoom,
|
|
|
|
'tab-change': top_window.tab_change,
|
|
|
|
'group-all': top_window.group_all,
|
|
|
|
'ungroup-all': top_window.ungroup_all,
|
|
|
|
'group-tab': top_window.group_tab,
|
|
|
|
'ungroup-tab': top_window.ungroup_tab,
|
2010-01-24 22:15:54 +00:00
|
|
|
'move-tab': top_window.move_tab,
|
2010-01-28 12:49:38 +00:00
|
|
|
'maximise': [top_window.zoom, False],
|
2010-04-02 15:45:32 +00:00
|
|
|
'tab-new': [top_window.tab_new, widget],
|
2010-01-29 13:12:33 +00:00
|
|
|
'navigate': top_window.navigate_terminal}
|
2009-11-14 22:58:22 +00:00
|
|
|
|
|
|
|
for signal in signals:
|
2010-01-24 12:55:03 +00:00
|
|
|
args = []
|
|
|
|
handler = signals[signal]
|
|
|
|
if isinstance(handler, list):
|
|
|
|
args = handler[1:]
|
|
|
|
handler = handler[0]
|
|
|
|
self.connect_child(widget, signal, handler, *args)
|
2009-11-14 22:58:22 +00:00
|
|
|
|
2009-11-14 23:04:15 +00:00
|
|
|
widget.grab_focus()
|
|
|
|
|
2009-11-08 23:06:26 +00:00
|
|
|
elif isinstance(widget, gtk.Paned):
|
2009-11-14 18:55:07 +00:00
|
|
|
try:
|
2009-11-14 22:58:22 +00:00
|
|
|
self.connect_child(widget, 'resize-term', self.resizeterm)
|
2009-11-14 18:55:07 +00:00
|
|
|
except TypeError:
|
|
|
|
err('Paned::add: %s has no signal resize-term' % widget)
|
2009-11-07 01:40:43 +00:00
|
|
|
|
|
|
|
def remove(self, widget):
|
|
|
|
"""Remove a widget from the container"""
|
|
|
|
gtk.Paned.remove(self, widget)
|
2009-11-14 22:58:22 +00:00
|
|
|
self.disconnect_child(widget)
|
2009-11-07 01:40:43 +00:00
|
|
|
self.children.remove(widget)
|
|
|
|
return(True)
|
|
|
|
|
2010-03-10 22:51:33 +00:00
|
|
|
def get_children(self):
|
|
|
|
"""Return an ordered list of our children"""
|
|
|
|
children = []
|
|
|
|
children.append(self.get_child1())
|
|
|
|
children.append(self.get_child2())
|
|
|
|
return(children)
|
|
|
|
|
2009-11-07 01:40:43 +00:00
|
|
|
def wrapcloseterm(self, widget):
|
|
|
|
"""A child terminal has closed, so this container must die"""
|
2009-12-09 13:02:31 +00:00
|
|
|
dbg('Paned::wrapcloseterm: Called on %s' % widget)
|
2009-11-07 01:40:43 +00:00
|
|
|
if self.closeterm(widget):
|
|
|
|
# At this point we only have one child, which is the surviving term
|
|
|
|
sibling = self.children[0]
|
|
|
|
self.remove(sibling)
|
2009-11-25 00:37:29 +00:00
|
|
|
|
|
|
|
parent = self.get_parent()
|
2009-12-09 13:02:31 +00:00
|
|
|
parent.remove(self)
|
2010-01-25 12:55:38 +00:00
|
|
|
self.cnxids.remove_all()
|
2009-12-09 13:02:31 +00:00
|
|
|
parent.add(sibling)
|
2009-11-07 01:40:43 +00:00
|
|
|
del(self)
|
|
|
|
else:
|
2009-11-14 18:55:07 +00:00
|
|
|
dbg("Paned::wrapcloseterm: self.closeterm failed")
|
2009-11-07 01:40:43 +00:00
|
|
|
|
2009-11-20 15:30:50 +00:00
|
|
|
def hoover(self):
|
|
|
|
"""Check that we still have a reason to exist"""
|
|
|
|
if len(self.children) == 1:
|
|
|
|
dbg('Paned::hoover: We only have one child, die')
|
|
|
|
parent = self.get_parent()
|
|
|
|
parent.remove(self)
|
|
|
|
child = self.children[0]
|
|
|
|
self.remove(child)
|
|
|
|
parent.add(child)
|
|
|
|
del(self)
|
|
|
|
|
2009-11-07 01:40:43 +00:00
|
|
|
def resizeterm(self, widget, keyname):
|
|
|
|
"""Handle a keyboard event requesting a terminal resize"""
|
2009-11-25 00:37:29 +00:00
|
|
|
maker = Factory()
|
2009-11-08 23:06:26 +00:00
|
|
|
if keyname in ['up', 'down'] and isinstance(self, gtk.VPaned):
|
|
|
|
# This is a key we can handle
|
|
|
|
position = self.get_position()
|
|
|
|
|
2009-11-25 00:37:29 +00:00
|
|
|
if maker.isinstance(widget, 'Terminal'):
|
2009-11-08 23:06:26 +00:00
|
|
|
fontheight = widget.vte.get_char_height()
|
|
|
|
else:
|
|
|
|
fontheight = 10
|
|
|
|
|
|
|
|
if keyname == 'up':
|
|
|
|
self.set_position(position - fontheight)
|
|
|
|
else:
|
|
|
|
self.set_position(position + fontheight)
|
|
|
|
elif keyname in ['left', 'right'] and isinstance(self, gtk.HPaned):
|
|
|
|
# This is a key we can handle
|
|
|
|
position = self.get_position()
|
|
|
|
|
2009-11-25 00:37:29 +00:00
|
|
|
if maker.isinstance(widget, 'Terminal'):
|
2009-11-08 23:06:26 +00:00
|
|
|
fontwidth = widget.vte.get_char_width()
|
|
|
|
else:
|
|
|
|
fontwidth = 10
|
|
|
|
|
|
|
|
if keyname == 'left':
|
|
|
|
self.set_position(position - fontwidth)
|
|
|
|
else:
|
|
|
|
self.set_position(position + fontwidth)
|
|
|
|
else:
|
|
|
|
# This is not a key we can handle
|
|
|
|
self.emit('resize-term', keyname)
|
2009-11-07 01:40:43 +00:00
|
|
|
|
2010-02-01 12:11:44 +00:00
|
|
|
def create_layout(self, layout):
|
|
|
|
"""Apply layout configuration"""
|
|
|
|
if not layout.has_key('children'):
|
|
|
|
err('layout specifies no children: %s' % layout)
|
|
|
|
return
|
|
|
|
|
|
|
|
children = layout['children']
|
|
|
|
if len(children) != 2:
|
|
|
|
# Paned widgets can only have two children
|
|
|
|
err('incorrect number of children for Paned: %s' % layout)
|
|
|
|
return
|
|
|
|
|
2010-03-11 13:04:01 +00:00
|
|
|
keys = []
|
|
|
|
|
|
|
|
# FIXME: This seems kinda ugly. All we want here is to know the order
|
|
|
|
# of children based on child['order']
|
|
|
|
try:
|
|
|
|
child_order_map = {}
|
|
|
|
for child in children:
|
|
|
|
key = children[child]['order']
|
|
|
|
child_order_map[key] = child
|
|
|
|
map_keys = child_order_map.keys()
|
|
|
|
map_keys.sort()
|
|
|
|
for map_key in map_keys:
|
|
|
|
keys.append(child_order_map[map_key])
|
|
|
|
except KeyError:
|
|
|
|
# We've failed to figure out the order. At least give the terminals
|
|
|
|
# in the wrong order
|
|
|
|
keys = children.keys()
|
|
|
|
|
2010-02-02 00:39:41 +00:00
|
|
|
num = 0
|
2010-02-27 13:04:50 +00:00
|
|
|
for child_key in keys:
|
2010-02-02 00:39:41 +00:00
|
|
|
child = children[child_key]
|
2010-02-01 12:11:44 +00:00
|
|
|
if child['type'] == 'Terminal':
|
2010-03-11 13:04:01 +00:00
|
|
|
pass
|
2010-02-01 12:11:44 +00:00
|
|
|
elif child['type'] == 'VPaned':
|
|
|
|
if num == 0:
|
|
|
|
terminal = self.get_child1()
|
|
|
|
else:
|
|
|
|
terminal = self.get_child2()
|
|
|
|
self.split_axis(terminal, True)
|
|
|
|
elif child['type'] == 'HPaned':
|
|
|
|
if num == 0:
|
|
|
|
terminal = self.get_child1()
|
|
|
|
else:
|
|
|
|
terminal = self.get_child2()
|
|
|
|
self.split_axis(terminal, False)
|
|
|
|
else:
|
|
|
|
err('unknown child type: %s' % child['type'])
|
2010-02-02 00:39:41 +00:00
|
|
|
num = num + 1
|
2010-02-01 12:11:44 +00:00
|
|
|
|
2010-02-02 00:39:41 +00:00
|
|
|
self.get_child1().create_layout(children[keys[0]])
|
|
|
|
self.get_child2().create_layout(children[keys[1]])
|
2010-02-01 12:11:44 +00:00
|
|
|
|
2010-02-27 13:55:38 +00:00
|
|
|
# FIXME: We need a delayed call to set_position, probably on realizing
|
|
|
|
# this widget, but it probably needs to start at the deepest widget and
|
|
|
|
# work back up. Fun.
|
|
|
|
|
2009-11-07 01:40:43 +00:00
|
|
|
class HPaned(Paned, gtk.HPaned):
|
|
|
|
"""Merge gtk.HPaned into our base Paned Container"""
|
|
|
|
def __init__(self):
|
|
|
|
"""Class initialiser"""
|
|
|
|
Paned.__init__(self)
|
|
|
|
gtk.HPaned.__init__(self)
|
2009-11-14 18:55:07 +00:00
|
|
|
self.register_signals(HPaned)
|
2010-01-18 13:17:35 +00:00
|
|
|
self.cnxids.new(self, 'expose-event', self.set_initial_position)
|
2009-11-07 01:40:43 +00:00
|
|
|
|
|
|
|
class VPaned(Paned, gtk.VPaned):
|
|
|
|
"""Merge gtk.VPaned into our base Paned Container"""
|
|
|
|
def __init__(self):
|
|
|
|
"""Class initialiser"""
|
|
|
|
Paned.__init__(self)
|
|
|
|
gtk.VPaned.__init__(self)
|
2009-11-14 18:55:07 +00:00
|
|
|
self.register_signals(VPaned)
|
2010-01-18 13:17:35 +00:00
|
|
|
self.cnxids.new(self, 'expose-event', self.set_initial_position)
|
2009-11-07 01:40:43 +00:00
|
|
|
|
|
|
|
gobject.type_register(HPaned)
|
|
|
|
gobject.type_register(VPaned)
|
|
|
|
# vim: set expandtab ts=4 sw=4:
|