8 import util.misc as misc_utils
9 import util.time_utils as time_utils
16 _moduleLogger = logging.getLogger(__name__)
19 class RadioWindow(windows._base.BasicWindow):
21 def __init__(self, app, player, store, node):
22 windows._base.BasicWindow.__init__(self, app, player, store)
24 self._childNode = None
26 self.connect_auto(self._player, "state-change", self._on_player_state_change)
27 self.connect_auto(self._player, "title-change", self._on_player_title_change)
29 self._loadingBanner = banners.GenericBanner()
31 headerPath = self._store.STORE_LOOKUP["radio_header"]
32 self._header = self._store.get_image_from_store(headerPath)
33 self._headerNavigation = presenter.NavigationBox()
34 self._headerNavigation.toplevel.add(self._header)
35 self._headerNavigation.connect("action", self._on_nav_action)
36 self._headerNavigation.connect("navigating", self._on_navigating)
38 self._programmingModel = gtk.ListStore(
43 textrenderer = gtk.CellRendererText()
44 timeColumn = gtk.TreeViewColumn("Time")
45 textrenderer.set_property("scale", 0.75)
46 timeColumn.set_property("sizing", gtk.TREE_VIEW_COLUMN_FIXED)
47 timeColumn.set_property("fixed-width", 80)
48 timeColumn.pack_start(textrenderer, expand=True)
49 timeColumn.add_attribute(textrenderer, "text", 0)
51 textrenderer = gtk.CellRendererText()
52 titleColumn = gtk.TreeViewColumn("Program")
53 titleColumn.set_property("sizing", gtk.TREE_VIEW_COLUMN_FIXED)
54 titleColumn.pack_start(textrenderer, expand=True)
55 titleColumn.add_attribute(textrenderer, "text", 1)
57 self._treeView = gtk.TreeView()
58 self._treeView.set_property("fixed-height-mode", True)
59 self._treeView.set_headers_visible(False)
60 self._treeView.set_model(self._programmingModel)
61 self._treeView.append_column(timeColumn)
62 self._treeView.append_column(titleColumn)
63 self._treeView.get_selection().connect("changed", self._on_row_changed)
65 viewport = gtk.Viewport()
66 viewport.add(self._treeView)
68 self._treeScroller = gtk.ScrolledWindow()
69 self._treeScroller.add(viewport)
70 self._treeScroller.set_policy(gtk.POLICY_NEVER, gtk.POLICY_AUTOMATIC)
71 self._treeScroller = hildonize.hildonize_scrollwindow(self._treeScroller)
73 self._presenter = presenter.StreamMiniPresenter(self._store)
74 self._presenterNavigation = presenter.NavigationBox()
75 self._presenterNavigation.toplevel.add(self._presenter.toplevel)
76 self._presenterNavigation.connect("action", self._on_nav_action)
77 self._presenterNavigation.connect("navigating", self._on_navigating)
79 self._radioLayout = gtk.VBox(False)
80 self._radioLayout.pack_start(self._headerNavigation.toplevel, False, False)
81 self._radioLayout.pack_start(self._treeScroller, True, True)
82 self._radioLayout.pack_start(self._presenterNavigation.toplevel, False, True)
84 self._layout.pack_start(self._loadingBanner.toplevel, False, False)
85 self._layout.pack_start(self._radioLayout, True, True)
87 self._dateShown = datetime.datetime.now(tz=time_utils.Mountain)
91 windows._base.BasicWindow.show(self)
93 self._errorBanner.toplevel.hide()
94 self._loadingBanner.toplevel.hide()
98 def jump_to(self, node):
99 _moduleLogger.info("Only 1 channel, nothing to jump to")
101 def _update_title(self):
102 self._window.set_title("%s - %s" % (self._node.title, self._dateShown.strftime("%m/%d")))
106 return self._player.node is self._childNode
108 def _set_context(self, state):
109 if state == self._player.STATE_PLAY:
111 self._presenter.set_state(self._store.STORE_LOOKUP["pause"])
113 self._presenter.set_state(self._store.STORE_LOOKUP["play"])
114 elif state == self._player.STATE_PAUSE:
115 self._presenter.set_state(self._store.STORE_LOOKUP["play"])
116 elif state == self._player.STATE_STOP:
117 self._presenter.set_state(self._store.STORE_LOOKUP["play"])
119 _moduleLogger.info("Unhandled player state %s" % state)
120 self._presenter.set_state(self._store.STORE_LOOKUP["play"])
122 def _show_loading(self):
123 animationPath = self._store.STORE_LOOKUP["loading"]
124 animation = self._store.get_pixbuf_animation_from_store(animationPath)
125 self._loadingBanner.show(animation, "Loading...")
127 def _hide_loading(self):
128 self._loadingBanner.hide()
132 self._programmingModel.clear()
133 self._node.get_children(
137 self._set_context(self._player.state)
139 def _get_current_row(self):
140 nowTime = self._dateShown.strftime("%H:%M:%S")
142 for i, row in enumerate(self._programmingModel):
151 @misc_utils.log_exception(_moduleLogger)
152 def _on_player_state_change(self, player, newState):
153 if self._headerNavigation.is_active() or self._presenterNavigation.is_active():
156 self._set_context(newState)
158 @misc_utils.log_exception(_moduleLogger)
159 def _on_player_title_change(self, player, node):
160 if node is not self._childNode or node is None:
161 _moduleLogger.info("Player title magically changed to %s" % player.title)
164 @misc_utils.log_exception(_moduleLogger)
165 def _on_navigating(self, widget, navState):
166 if navState == "clicking":
167 if self._player.state == self._player.STATE_PLAY:
169 imageName = "pause_pressed"
171 imageName = "play_pressed"
172 elif self._player.state == self._player.STATE_PAUSE:
173 imageName = "play_pressed"
174 elif self._player.state == self._player.STATE_STOP:
175 imageName = "play_pressed"
177 imageName = "play_pressed"
178 _moduleLogger.info("Unhandled player state %s" % self._player.state)
179 elif navState == "down":
182 if self._player.state == self._player.STATE_PLAY:
187 self._presenter.set_state(self._store.STORE_LOOKUP[imageName])
189 @misc_utils.log_exception(_moduleLogger)
190 def _on_nav_action(self, widget, navState):
191 self._set_context(self._player.state)
193 if navState == "clicking":
194 if self._player.state == self._player.STATE_PLAY:
198 self._player.set_piece_by_node(self._childNode)
200 elif self._player.state == self._player.STATE_PAUSE:
202 elif self._player.state == self._player.STATE_STOP:
203 self._player.set_piece_by_node(self._childNode)
206 _moduleLogger.info("Unhandled player state %s" % self._player.state)
207 elif navState == "down":
208 self.window.destroy()
209 elif navState == "up":
211 elif navState == "left":
212 self._dateShown += datetime.timedelta(days=1)
215 elif navState == "right":
216 self._dateShown -= datetime.timedelta(days=1)
220 @misc_utils.log_exception(_moduleLogger)
221 def _on_channels(self, channels):
222 if self._isDestroyed:
223 _moduleLogger.info("Download complete but window destroyed")
227 if 1 < len(channels):
228 _moduleLogger.warning("More channels now available!")
229 self._childNode = channels[0]
230 self._childNode.get_programming(
236 @misc_utils.log_exception(_moduleLogger)
237 def _on_channel(self, programs):
238 if self._isDestroyed:
239 _moduleLogger.info("Download complete but window destroyed")
243 for program in programs:
244 row = program["time"], program["title"]
245 self._programmingModel.append(row)
247 currentDate = datetime.datetime.now()
248 if currentDate.date() != self._dateShown.date():
249 self._treeView.get_selection().set_mode(gtk.SELECTION_NONE)
251 self._treeView.get_selection().set_mode(gtk.SELECTION_SINGLE)
252 path = (self._get_current_row(), )
253 self._treeView.scroll_to_cell(path)
254 self._treeView.get_selection().select_path(path)
256 @misc_utils.log_exception(_moduleLogger)
257 def _on_load_error(self, exception):
259 self._errorBanner.push_message(str(exception))
261 @misc_utils.log_exception(_moduleLogger)
262 def _on_row_changed(self, selection):
263 if len(self._programmingModel) == 0:
266 rowIndex = self._get_current_row()
268 if not selection.path_is_selected(path):
269 # Undo the user's changing of the selection
270 selection.select_path(path)
273 gobject.type_register(RadioWindow)