1 import gtk
, gobject
, os
2 from zeroinstall
.injector
import model
, writer
3 from gui
import policy
, pretty_size
4 from treetips
import TreeTips
6 def popup_menu(bev
, values
, fn
):
10 item
= gtk
.SeparatorMenuItem()
12 item
= gtk
.MenuItem(str(value
).capitalize())
13 item
.connect('activate', lambda item
, v
=value
: fn(v
))
16 menu
.popup(None, None, None, bev
.button
, bev
.time
)
18 rox_filer
= 'http://rox.sourceforge.net/2005/interfaces/ROX-Filer'
29 def interface_for(interface
, impl
):
30 # Since injector 0.16, we can get the interface from the impl
31 # Before that, the impl always comes from the main interface (even for feeds)
32 if hasattr(impl
, 'interface'):
36 class ImplTips(TreeTips
):
37 def get_tooltip_text(self
, impl
):
38 if impl
.id.startswith('/'):
39 return _("Local: %s") % impl
.id
40 if policy
.get_cached(impl
):
41 return _("Cached: %s") % policy
.get_implementation_path(impl
)
43 src
= policy
.get_best_source(impl
)
45 size
= pretty_size(src
.size
)
46 return _("Not yet downloaded (%s)") % size
48 return _("No downloads available!")
52 class ImplementationList(gtk
.ScrolledWindow
):
56 def __init__(self
, interface
):
57 gtk
.ScrolledWindow
.__init
__(self
, None, None)
58 self
.set_shadow_type(gtk
.SHADOW_IN
)
60 self
.model
= gtk
.ListStore(object, str, str, str,
61 gobject
.TYPE_BOOLEAN
, gobject
.TYPE_BOOLEAN
,
64 self
.tree_view
= gtk
.TreeView(self
.model
)
66 text
= gtk
.CellRendererText()
67 text_strike
= gtk
.CellRendererText()
68 toggle
= gtk
.CellRendererToggle()
70 stability
= gtk
.TreeViewColumn('Stability', text
, text
= STABILITY
)
72 for column
in (gtk
.TreeViewColumn('Version', text
, text
= VERSION
, strikethrough
= UNUSABLE
),
73 gtk
.TreeViewColumn('Released', text
, text
= RELEASED
, strikethrough
= UNUSABLE
),
75 gtk
.TreeViewColumn('C', toggle
, active
= CACHED
),
76 gtk
.TreeViewColumn('Arch', text
, text
= ARCH
)):
77 self
.tree_view
.append_column(column
)
79 self
.add(self
.tree_view
)
81 def motion(tree_view
, ev
):
82 if ev
.window
is not tree_view
.get_bin_window():
84 pos
= tree_view
.get_path_at_pos(int(ev
.x
), int(ev
.y
))
87 row
= self
.model
[path
]
88 if row
[ITEM
] is not tips
.item
:
89 tips
.prime(tree_view
, row
[ITEM
])
93 self
.tree_view
.connect('motion-notify-event', motion
)
94 self
.tree_view
.connect('leave-notify-event', lambda tv
, ev
: tips
.hide())
95 self
.tree_view
.connect('destroy', lambda tv
: tips
.hide())
97 def button_press(tree_view
, bev
):
98 if bev
.button
not in (1, 3):
100 pos
= tree_view
.get_path_at_pos(int(bev
.x
), int(bev
.y
))
103 path
, col
, x
, y
= pos
104 impl
= self
.model
[path
][ITEM
]
106 upstream
= impl
.upstream_stability
or model
.testing
107 choices
= model
.stability_levels
.values()
111 if isinstance(new
, model
.Stability
):
112 impl
.user_stability
= new
114 impl
.user_stability
= None
115 writer
.save_interface(interface_for(interface
, impl
))
117 popup_menu(bev
, ['Unset (%s)' % upstream
, None] + choices
,
119 elif bev
.button
== 3 and policy
.get_cached(impl
):
121 os
.spawnlp(os
.P_WAIT
, '0launch',
122 '0launch', rox_filer
, '-d',
123 policy
.get_implementation_path(impl
))
124 popup_menu(bev
, ['Open cached copy'], open)
125 self
.tree_view
.connect('button-press-event', button_press
)
127 def get_selection(self
):
128 return self
.tree_view
.get_selection()
130 def set_items(self
, items
):
133 new
= self
.model
.append()
134 self
.model
[new
][ITEM
] = item
135 self
.model
[new
][VERSION
] = item
.get_version()
136 if hasattr(item
, 'released') and item
.released
:
137 self
.model
[new
][RELEASED
] = item
.released
139 self
.model
[new
][RELEASED
] = "-"
140 self
.model
[new
][CACHED
] = policy
.get_cached(item
)
141 if item
.user_stability
:
142 self
.model
[new
][STABILITY
] = str(item
.user_stability
).upper()
144 self
.model
[new
][STABILITY
] = item
.upstream_stability
or \
146 self
.model
[new
][ARCH
] = item
.arch
or 'any'
147 self
.model
[new
][UNUSABLE
] = policy
.is_unusable(item
)