1 # @todo Add icons to buttons/rows to indicate that the currently playing track is coming from that
12 import util.misc as misc_utils
13 import util.go_utils as go_utils
21 _moduleLogger = logging.getLogger(__name__)
24 class BasicWindow(gobject.GObject, go_utils.AutoSignal):
28 gobject.SIGNAL_RUN_LAST,
33 gobject.SIGNAL_RUN_LAST,
38 gobject.SIGNAL_RUN_LAST,
40 (gobject.TYPE_PYOBJECT, ),
43 gobject.SIGNAL_RUN_LAST,
45 (gobject.TYPE_BOOLEAN, ),
48 gobject.SIGNAL_RUN_LAST,
50 (gobject.TYPE_BOOLEAN, ),
54 def __init__(self, player, store):
55 gobject.GObject.__init__(self)
56 self._isDestroyed = False
61 self._clipboard = gtk.clipboard_get()
62 self._windowInFullscreen = False
64 self._errorBanner = banners.StackingBanner()
66 self._layout = gtk.VBox()
67 self._layout.pack_start(self._errorBanner.toplevel, False, True)
69 self._window = gtk.Window()
70 go_utils.AutoSignal.__init__(self, self.window)
71 self._window.add(self._layout)
72 self._window = hildonize.hildonize_window(self, self._window)
74 self._window.set_icon(self._store.get_pixbuf_from_store(self._store.STORE_LOOKUP["icon"]))
75 self._window.connect("key-press-event", self._on_key_press)
76 self._window.connect("window-state-event", self._on_window_state_change)
77 self._window.connect("destroy", self._on_destroy)
84 self._window.show_all()
86 def save_settings(self, config, sectionName):
87 config.add_section(sectionName)
88 config.set(sectionName, "fullscreen", str(self._windowInFullscreen))
90 def load_settings(self, config, sectionName):
92 self._windowInFullscreen = config.getboolean(sectionName, "fullscreen")
93 except ConfigParser.NoSectionError, e:
95 "Settings file %s is missing section %s" % (
96 constants._user_settings_,
101 if self._windowInFullscreen:
102 self._window.fullscreen()
104 self._window.unfullscreen()
106 def jump_to(self, node):
107 raise NotImplementedError("On %s" % self)
109 @misc_utils.log_exception(_moduleLogger)
110 def _on_destroy(self, *args):
111 self._isDestroyed = True
113 @misc_utils.log_exception(_moduleLogger)
114 def _on_window_state_change(self, widget, event, *args):
115 if event.new_window_state & gtk.gdk.WINDOW_STATE_FULLSCREEN:
116 self._windowInFullscreen = True
118 self._windowInFullscreen = False
119 self.emit("fullscreen", self._windowInFullscreen)
121 @misc_utils.log_exception(_moduleLogger)
122 def _on_key_press(self, widget, event, *args):
123 RETURN_TYPES = (gtk.keysyms.Return, gtk.keysyms.ISO_Enter, gtk.keysyms.KP_Enter)
124 isCtrl = bool(event.get_state() & gtk.gdk.CONTROL_MASK)
126 event.keyval == gtk.keysyms.F6 or
127 event.keyval in RETURN_TYPES and isCtrl
129 # The "Full screen" hardware key has been pressed
130 if self._windowInFullscreen:
131 self._window.unfullscreen ()
133 self._window.fullscreen ()
136 event.keyval in (gtk.keysyms.w, ) and
137 event.get_state() & gtk.gdk.CONTROL_MASK
139 self._window.destroy()
141 event.keyval in (gtk.keysyms.q, ) and
142 event.get_state() & gtk.gdk.CONTROL_MASK
145 elif event.keyval == gtk.keysyms.l and event.get_state() & gtk.gdk.CONTROL_MASK:
146 with open(constants._user_logpath_, "r") as f:
147 logLines = f.xreadlines()
148 log = "".join(logLines)
149 self._clipboard.set_text(str(log))
152 @misc_utils.log_exception(_moduleLogger)
153 def _on_home(self, *args):
155 self._window.destroy()
157 @misc_utils.log_exception(_moduleLogger)
158 def _on_jump(self, source, node):
159 raise NotImplementedError("On %s" % self)
161 @misc_utils.log_exception(_moduleLogger)
162 def _on_quit(self, *args):
164 self._window.destroy()
167 class SourceSelector(BasicWindow):
169 def __init__(self, player, store, index):
170 BasicWindow.__init__(self, player, store)
174 self._loadingBanner = banners.GenericBanner()
176 self._radioButton = self._create_button("radio", "Radio")
177 self._radioButton.connect("clicked", self._on_source_selected, stream_index.SOURCE_RADIO)
178 self._radioWrapper = gtk.VBox()
179 self._radioWrapper.pack_start(self._radioButton, False, True)
181 self._conferenceButton = self._create_button("conferences", "Conferences")
182 self._conferenceButton.connect("clicked", self._on_source_selected, stream_index.SOURCE_CONFERENCES)
183 self._conferenceWrapper = gtk.VBox()
184 self._conferenceWrapper.pack_start(self._conferenceButton, False, True)
186 self._magazineButton = self._create_button("magazines", "Magazines")
187 self._magazineButton.connect("clicked", self._on_source_selected, stream_index.SOURCE_MAGAZINES)
188 self._magazineWrapper = gtk.VBox()
189 self._magazineWrapper.pack_start(self._magazineButton, False, True)
191 self._scriptureButton = self._create_button("scriptures", "Scriptures")
192 self._scriptureButton.connect("clicked", self._on_source_selected, stream_index.SOURCE_SCRIPTURES)
193 self._scriptureWrapper = gtk.VBox()
194 self._scriptureWrapper.pack_start(self._scriptureButton, False, True)
196 self._buttonLayout = gtk.VButtonBox()
197 self._buttonLayout.set_layout(gtk.BUTTONBOX_SPREAD)
198 self._buttonLayout.pack_start(self._radioWrapper, True, True)
199 self._buttonLayout.pack_start(self._conferenceWrapper, True, True)
200 self._buttonLayout.pack_start(self._magazineWrapper, True, True)
201 self._buttonLayout.pack_start(self._scriptureWrapper, True, True)
203 self._separator = gtk.HSeparator()
204 self._playcontrol = playcontrol.NavControl(player, store)
205 self._playcontrol.connect("jump-to", self._on_jump)
207 self._layout.pack_start(self._loadingBanner.toplevel, False, False)
208 self._layout.pack_start(self._buttonLayout, True, True)
209 self._layout.pack_start(self._separator, False, True)
210 self._layout.pack_start(self._playcontrol.toplevel, False, True)
212 self._window.set_title(constants.__pretty_app_name__)
215 BasicWindow.show(self)
217 self._errorBanner.toplevel.hide()
218 self._playcontrol.toplevel.hide()
222 def _show_loading(self):
223 animationPath = self._store.STORE_LOOKUP["loading"]
224 animation = self._store.get_pixbuf_animation_from_store(animationPath)
225 self._loadingBanner.show(animation, "Loading...")
226 self._buttonLayout.set_sensitive(False)
228 def _hide_loading(self):
229 self._loadingBanner.hide()
230 self._buttonLayout.set_sensitive(True)
234 self._index.get_languages(self._on_languages, self._on_error)
236 def _create_button(self, icon, message):
237 image = self._store.get_image_from_store(self._store.STORE_LOOKUP[icon])
240 label.set_text(message)
242 buttonLayout = gtk.HBox(False, 5)
243 buttonLayout.pack_start(image, False, False)
244 buttonLayout.pack_start(label, False, True)
245 button = gtk.Button()
246 button.add(buttonLayout)
250 @misc_utils.log_exception(_moduleLogger)
251 def _on_languages(self, languages):
253 self._languages = list(languages)
255 @misc_utils.log_exception(_moduleLogger)
256 def _on_error(self, exception):
258 self._errorBanner.push_message(str(exception))
260 def _window_from_node(self, node):
261 if node.id == stream_index.SOURCE_RADIO:
263 elif node.id == stream_index.SOURCE_CONFERENCES:
264 Source = ConferencesWindow
265 elif node.id == stream_index.SOURCE_MAGAZINES:
267 elif node.id == stream_index.SOURCE_SCRIPTURES:
269 sourceWindow = Source(self._player, self._store, node)
270 sourceWindow.window.set_modal(True)
271 sourceWindow.window.set_transient_for(self._window)
272 sourceWindow.window.set_default_size(*self._window.get_size())
273 sourceWindow.connect("quit", self._on_quit)
274 sourceWindow.connect("jump-to", self._on_jump)
278 @misc_utils.log_exception(_moduleLogger)
279 def _on_jump(self, source, node):
280 targetNodePath = list(reversed(list(stream_index.walk_ancestors(node))))
281 ancestor = targetNodePath[0]
282 window = self._window_from_node(ancestor)
285 @misc_utils.log_exception(_moduleLogger)
286 def _on_source_selected(self, widget, nodeName):
287 node = self._index.get_source(nodeName, self._languages[0]["id"])
288 self._window_from_node(node)
291 gobject.type_register(SourceSelector)
294 class RadioWindow(BasicWindow):
296 def __init__(self, player, store, node):
297 BasicWindow.__init__(self, player, store)
299 self._childNode = None
301 self.connect_auto(self._player, "state-change", self._on_player_state_change)
302 self.connect_auto(self._player, "title-change", self._on_player_title_change)
304 self._loadingBanner = banners.GenericBanner()
306 headerPath = self._store.STORE_LOOKUP["radio_header"]
307 self._header = self._store.get_image_from_store(headerPath)
308 self._headerNavigation = presenter.NavigationBox()
309 self._headerNavigation.toplevel.add(self._header)
310 self._headerNavigation.connect("action", self._on_nav_action)
311 self._headerNavigation.connect("navigating", self._on_navigating)
313 self._programmingModel = gtk.ListStore(
318 textrenderer = gtk.CellRendererText()
319 timeColumn = gtk.TreeViewColumn("Time")
320 timeColumn.pack_start(textrenderer, expand=True)
321 timeColumn.add_attribute(textrenderer, "text", 0)
323 textrenderer = gtk.CellRendererText()
324 titleColumn = gtk.TreeViewColumn("Program")
325 titleColumn.pack_start(textrenderer, expand=True)
326 titleColumn.add_attribute(textrenderer, "text", 1)
328 self._treeView = gtk.TreeView()
329 self._treeView.set_headers_visible(False)
330 self._treeView.set_model(self._programmingModel)
331 self._treeView.append_column(timeColumn)
332 self._treeView.append_column(titleColumn)
333 self._treeView.get_selection().connect("changed", self._on_row_changed)
335 self._treeScroller = gtk.ScrolledWindow()
336 self._treeScroller.add(self._treeView)
337 self._treeScroller.set_policy(gtk.POLICY_NEVER, gtk.POLICY_AUTOMATIC)
339 self._presenter = presenter.StreamMiniPresenter(self._store)
340 self._presenterNavigation = presenter.NavigationBox()
341 self._presenterNavigation.toplevel.add(self._presenter.toplevel)
342 self._presenterNavigation.connect("action", self._on_nav_action)
343 self._presenterNavigation.connect("navigating", self._on_navigating)
345 self._radioLayout = gtk.VBox(False)
346 self._radioLayout.pack_start(self._headerNavigation.toplevel, False, False)
347 self._radioLayout.pack_start(self._treeScroller, True, True)
348 self._radioLayout.pack_start(self._presenterNavigation.toplevel, False, True)
350 self._layout.pack_start(self._loadingBanner.toplevel, False, False)
351 self._layout.pack_start(self._radioLayout, True, True)
353 self._dateShown = datetime.datetime.now()
357 BasicWindow.show(self)
359 self._errorBanner.toplevel.hide()
360 self._loadingBanner.toplevel.hide()
364 def jump_to(self, node):
365 _moduleLogger.info("Only 1 channel, nothing to jump to")
367 def _update_title(self):
368 self._window.set_title("%s - %s" % (self._node.title, self._dateShown.strftime("%m/%d")))
372 return self._player.node is self._childNode
374 def _set_context(self, state):
375 if state == self._player.STATE_PLAY:
377 self._presenter.set_state(self._store.STORE_LOOKUP["pause"])
379 self._presenter.set_state(self._store.STORE_LOOKUP["play"])
380 elif state == self._player.STATE_PAUSE:
381 self._presenter.set_state(self._store.STORE_LOOKUP["play"])
382 elif state == self._player.STATE_STOP:
383 self._presenter.set_state(self._store.STORE_LOOKUP["play"])
385 _moduleLogger.info("Unhandled player state %s" % state)
386 self._presenter.set_state(self._store.STORE_LOOKUP["play"])
388 def _show_loading(self):
389 animationPath = self._store.STORE_LOOKUP["loading"]
390 animation = self._store.get_pixbuf_animation_from_store(animationPath)
391 self._loadingBanner.show(animation, "Loading...")
393 def _hide_loading(self):
394 self._loadingBanner.hide()
398 self._programmingModel.clear()
399 self._node.get_children(
403 self._set_context(self._player.state)
405 def _get_current_row(self):
406 nowTime = self._dateShown.strftime("%H:%M:%S")
408 for i, row in enumerate(self._programmingModel):
417 @misc_utils.log_exception(_moduleLogger)
418 def _on_player_state_change(self, player, newState):
419 if self._headerNavigation.is_active() or self._presenterNavigation.is_active():
422 self._set_context(newState)
424 @misc_utils.log_exception(_moduleLogger)
425 def _on_player_title_change(self, player, node):
426 if node is not self._childNode or node is None:
427 _moduleLogger.info("Player title magically changed to %s" % player.title)
430 @misc_utils.log_exception(_moduleLogger)
431 def _on_navigating(self, widget, navState):
432 if navState == "clicking":
433 if self._player.state == self._player.STATE_PLAY:
435 imageName = "pause_pressed"
437 imageName = "play_pressed"
438 elif self._player.state == self._player.STATE_PAUSE:
439 imageName = "play_pressed"
440 elif self._player.state == self._player.STATE_STOP:
441 imageName = "play_pressed"
443 imageName = "play_pressed"
444 _moduleLogger.info("Unhandled player state %s" % self._player.state)
445 elif navState == "down":
448 if self._player.state == self._player.STATE_PLAY:
453 self._presenter.set_state(self._store.STORE_LOOKUP[imageName])
455 @misc_utils.log_exception(_moduleLogger)
456 def _on_nav_action(self, widget, navState):
457 self._set_context(self._player.state)
459 if navState == "clicking":
460 if self._player.state == self._player.STATE_PLAY:
464 self._player.set_piece_by_node(self._childNode)
466 elif self._player.state == self._player.STATE_PAUSE:
468 elif self._player.state == self._player.STATE_STOP:
469 self._player.set_piece_by_node(self._childNode)
472 _moduleLogger.info("Unhandled player state %s" % self._player.state)
473 elif navState == "down":
474 self.window.destroy()
475 elif navState == "up":
477 elif navState == "left":
478 self._dateShown += datetime.timedelta(days=1)
481 elif navState == "right":
482 self._dateShown -= datetime.timedelta(days=1)
486 @misc_utils.log_exception(_moduleLogger)
487 def _on_channels(self, channels):
488 if self._isDestroyed:
489 _moduleLogger.info("Download complete but window destroyed")
493 if 1 < len(channels):
494 _moduleLogger.warning("More channels now available!")
495 self._childNode = channels[0]
496 self._childNode.get_programming(
502 @misc_utils.log_exception(_moduleLogger)
503 def _on_channel(self, programs):
504 if self._isDestroyed:
505 _moduleLogger.info("Download complete but window destroyed")
509 for program in programs:
510 row = program["time"], program["title"]
511 self._programmingModel.append(row)
513 currentDate = datetime.datetime.now()
514 if currentDate.date() != self._dateShown.date():
515 self._treeView.get_selection().set_mode(gtk.SELECTION_NONE)
517 self._treeView.get_selection().set_mode(gtk.SELECTION_SINGLE)
518 path = (self._get_current_row(), )
519 self._treeView.scroll_to_cell(path)
520 self._treeView.get_selection().select_path(path)
522 @misc_utils.log_exception(_moduleLogger)
523 def _on_load_error(self, exception):
525 self._errorBanner.push_message(str(exception))
527 @misc_utils.log_exception(_moduleLogger)
528 def _on_row_changed(self, selection):
529 if len(self._programmingModel) == 0:
532 rowIndex = self._get_current_row()
534 if not selection.path_is_selected(path):
535 # Undo the user's changing of the selection
536 selection.select_path(path)
539 gobject.type_register(RadioWindow)
542 class ListWindow(BasicWindow):
544 def __init__(self, player, store, node):
545 BasicWindow.__init__(self, player, store)
548 self.connect_auto(self._player, "title-change", self._on_player_title_change)
550 self._loadingBanner = banners.GenericBanner()
552 modelTypes, columns = zip(*self._get_columns())
554 self._model = gtk.ListStore(*modelTypes)
556 self._treeView = gtk.TreeView()
557 self._treeView.connect("row-activated", self._on_row_activated)
558 self._treeView.set_headers_visible(False)
559 self._treeView.set_model(self._model)
560 for column in columns:
561 if column is not None:
562 self._treeView.append_column(column)
564 self._treeScroller = gtk.ScrolledWindow()
565 self._treeScroller.add(self._treeView)
566 self._treeScroller.set_policy(gtk.POLICY_NEVER, gtk.POLICY_AUTOMATIC)
568 self._separator = gtk.HSeparator()
569 self._playcontrol = playcontrol.NavControl(self._player, self._store)
570 self._playcontrol.connect("home", self._on_home)
571 self._playcontrol.connect("jump-to", self._on_jump)
573 self._contentLayout = gtk.VBox(False)
574 self._contentLayout.pack_start(self._treeScroller, True, True)
575 self._contentLayout.pack_start(self._separator, False, True)
576 self._contentLayout.pack_start(self._playcontrol.toplevel, False, True)
578 self._layout.pack_start(self._loadingBanner.toplevel, False, False)
579 self._layout.pack_start(self._contentLayout, True, True)
582 BasicWindow.show(self)
584 self._errorBanner.toplevel.hide()
585 self._loadingBanner.toplevel.hide()
588 self._playcontrol.refresh()
591 def _get_columns(cls):
592 raise NotImplementedError("")
594 def _get_current_row(self):
595 if self._player.node is None:
597 ancestors, current, descendants = stream_index.common_paths(self._player.node, self._node)
600 activeChild = descendants[0]
601 for i, row in enumerate(self._model):
602 if activeChild is row[0]:
607 def jump_to(self, node):
608 ancestors, current, descendants = stream_index.common_paths(node, self._node)
610 raise RuntimeError("Cannot jump to node %s" % node)
612 _moduleLogger.info("Current node is the target")
614 child = descendants[0]
615 window = self._window_from_node(child)
618 def _window_from_node(self, node):
619 raise NotImplementedError("")
621 @misc_utils.log_exception(_moduleLogger)
622 def _on_row_activated(self, view, path, column):
623 raise NotImplementedError("")
625 @misc_utils.log_exception(_moduleLogger)
626 def _on_player_title_change(self, player, node):
629 @misc_utils.log_exception(_moduleLogger)
630 def _on_jump(self, source, node):
631 ancestors, current, descendants = stream_index.common_paths(node, self._node)
633 _moduleLogger.info("%s is not the target, moving up" % self._node)
634 self.emit("jump-to", node)
635 self._window.destroy()
638 _moduleLogger.info("Current node is the target")
640 child = descendants[0]
641 window = self._window_from_node(child)
644 def _show_loading(self):
645 animationPath = self._store.STORE_LOOKUP["loading"]
646 animation = self._store.get_pixbuf_animation_from_store(animationPath)
647 self._loadingBanner.show(animation, "Loading...")
649 def _hide_loading(self):
650 self._loadingBanner.hide()
656 def _select_row(self):
657 rowIndex = self._get_current_row()
661 self._treeView.scroll_to_cell(path)
662 self._treeView.get_selection().select_path(path)
665 class ConferencesWindow(ListWindow):
667 def __init__(self, player, store, node):
668 ListWindow.__init__(self, player, store, node)
669 self._window.set_title(self._node.title)
672 def _get_columns(cls):
673 yield gobject.TYPE_PYOBJECT, None
675 textrenderer = gtk.CellRendererText()
676 column = gtk.TreeViewColumn("Date")
677 column.pack_start(textrenderer, expand=True)
678 column.add_attribute(textrenderer, "text", 1)
679 yield gobject.TYPE_STRING, column
681 textrenderer = gtk.CellRendererText()
682 column = gtk.TreeViewColumn("Conference")
683 column.pack_start(textrenderer, expand=True)
684 column.add_attribute(textrenderer, "text", 2)
685 yield gobject.TYPE_STRING, column
688 ListWindow._refresh(self)
689 self._node.get_children(
690 self._on_conferences,
694 @misc_utils.log_exception(_moduleLogger)
695 def _on_conferences(self, programs):
696 if self._isDestroyed:
697 _moduleLogger.info("Download complete but window destroyed")
701 for programNode in programs:
702 program = programNode.get_properties()
703 row = programNode, program["title"], program["full_title"]
704 self._model.append(row)
708 @misc_utils.log_exception(_moduleLogger)
709 def _on_error(self, exception):
711 self._errorBanner.push_message(str(exception))
713 def _window_from_node(self, node):
714 sessionsWindow = ConferenceSessionsWindow(self._player, self._store, node)
715 sessionsWindow.window.set_modal(True)
716 sessionsWindow.window.set_transient_for(self._window)
717 sessionsWindow.window.set_default_size(*self._window.get_size())
718 sessionsWindow.connect("quit", self._on_quit)
719 sessionsWindow.connect("home", self._on_home)
720 sessionsWindow.connect("jump-to", self._on_jump)
721 sessionsWindow.show()
722 return sessionsWindow
724 @misc_utils.log_exception(_moduleLogger)
725 def _on_row_activated(self, view, path, column):
726 itr = self._model.get_iter(path)
727 node = self._model.get_value(itr, 0)
728 self._window_from_node(node)
731 gobject.type_register(ConferencesWindow)
734 class ConferenceSessionsWindow(ListWindow):
736 def __init__(self, player, store, node):
737 ListWindow.__init__(self, player, store, node)
738 self._window.set_title(self._node.title)
741 def _get_columns(cls):
742 yield gobject.TYPE_PYOBJECT, None
744 textrenderer = gtk.CellRendererText()
745 column = gtk.TreeViewColumn("Session")
746 column.pack_start(textrenderer, expand=True)
747 column.add_attribute(textrenderer, "text", 1)
748 yield gobject.TYPE_STRING, column
751 ListWindow._refresh(self)
752 self._node.get_children(
753 self._on_conference_sessions,
757 @misc_utils.log_exception(_moduleLogger)
758 def _on_conference_sessions(self, programs):
759 if self._isDestroyed:
760 _moduleLogger.info("Download complete but window destroyed")
764 for programNode in programs:
765 program = programNode.get_properties()
766 row = programNode, program["title"]
767 self._model.append(row)
771 @misc_utils.log_exception(_moduleLogger)
772 def _on_error(self, exception):
774 self._errorBanner.push_message(str(exception))
776 def _window_from_node(self, node):
777 sessionsWindow = ConferenceTalksWindow(self._player, self._store, node)
778 sessionsWindow.window.set_modal(True)
779 sessionsWindow.window.set_transient_for(self._window)
780 sessionsWindow.window.set_default_size(*self._window.get_size())
781 sessionsWindow.connect("quit", self._on_quit)
782 sessionsWindow.connect("home", self._on_home)
783 sessionsWindow.connect("jump-to", self._on_jump)
784 sessionsWindow.show()
785 return sessionsWindow
787 @misc_utils.log_exception(_moduleLogger)
788 def _on_row_activated(self, view, path, column):
789 itr = self._model.get_iter(path)
790 node = self._model.get_value(itr, 0)
791 self._window_from_node(node)
794 gobject.type_register(ConferenceSessionsWindow)
797 class ConferenceTalksWindow(ListWindow):
799 def __init__(self, player, store, node):
800 ListWindow.__init__(self, player, store, node)
801 self._window.set_title(self._node.title)
804 def _get_columns(cls):
805 yield gobject.TYPE_PYOBJECT, None
807 textrenderer = gtk.CellRendererText()
808 column = gtk.TreeViewColumn("Talk")
809 column.pack_start(textrenderer, expand=True)
810 column.add_attribute(textrenderer, "text", 1)
811 yield gobject.TYPE_STRING, column
814 ListWindow._refresh(self)
815 self._node.get_children(
816 self._on_conference_talks,
820 @misc_utils.log_exception(_moduleLogger)
821 def _on_conference_talks(self, programs):
822 if self._isDestroyed:
823 _moduleLogger.info("Download complete but window destroyed")
827 for programNode in programs:
828 program = programNode.get_properties()
829 row = programNode, "%s\n%s" % (program["title"], program["speaker"])
830 self._model.append(row)
834 @misc_utils.log_exception(_moduleLogger)
835 def _on_error(self, exception):
837 self._errorBanner.push_message(str(exception))
839 def _window_from_node(self, node):
840 sessionsWindow = ConferenceTalkWindow(self._player, self._store, node)
841 sessionsWindow.window.set_modal(True)
842 sessionsWindow.window.set_transient_for(self._window)
843 sessionsWindow.window.set_default_size(*self._window.get_size())
844 sessionsWindow.connect("quit", self._on_quit)
845 sessionsWindow.connect("home", self._on_home)
846 sessionsWindow.connect("jump-to", self._on_jump)
847 sessionsWindow.show()
848 return sessionsWindow
850 @misc_utils.log_exception(_moduleLogger)
851 def _on_row_activated(self, view, path, column):
852 itr = self._model.get_iter(path)
853 node = self._model.get_value(itr, 0)
854 self._window_from_node(node)
857 gobject.type_register(ConferenceTalksWindow)
860 class ConferenceTalkWindow(BasicWindow):
862 def __init__(self, player, store, node):
863 BasicWindow.__init__(self, player, store)
865 self._playerNode = self._player.node
866 self._nextSearch = None
867 self._updateSeek = None
869 self.connect_auto(self._player, "state-change", self._on_player_state_change)
870 self.connect_auto(self._player, "title-change", self._on_player_title_change)
871 self.connect_auto(self._player, "error", self._on_player_error)
873 self._loadingBanner = banners.GenericBanner()
875 self._presenter = presenter.StreamPresenter(self._store)
876 self._presenter.set_context(
877 self._store.STORE_LOOKUP["conference_background"],
881 self._presenterNavigation = presenter.NavigationBox()
882 self._presenterNavigation.toplevel.add(self._presenter.toplevel)
883 self._presenterNavigation.connect("action", self._on_nav_action)
884 self._presenterNavigation.connect("navigating", self._on_navigating)
886 self._seekbar = hildonize.create_seekbar()
887 self._seekbar.connect("change-value", self._on_user_seek)
889 self._layout.pack_start(self._loadingBanner.toplevel, False, False)
890 self._layout.pack_start(self._presenterNavigation.toplevel, True, True)
891 self._layout.pack_start(self._seekbar, False, False)
893 self._window.set_title(self._node.title)
896 BasicWindow.show(self)
897 self._window.show_all()
898 self._errorBanner.toplevel.hide()
899 self._loadingBanner.toplevel.hide()
900 self._set_context(self._player.state)
903 def jump_to(self, node):
904 assert self._node is node
908 return self._playerNode is self._node
910 def _show_loading(self):
911 animationPath = self._store.STORE_LOOKUP["loading"]
912 animation = self._store.get_pixbuf_animation_from_store(animationPath)
913 self._loadingBanner.show(animation, "Loading...")
915 def _hide_loading(self):
916 self._loadingBanner.hide()
918 def _set_context(self, state):
919 if state == self._player.STATE_PLAY:
921 self._presenter.set_state(self._store.STORE_LOOKUP["pause"])
923 self._presenter.set_state(self._store.STORE_LOOKUP["play"])
924 elif state == self._player.STATE_PAUSE:
925 self._presenter.set_state(self._store.STORE_LOOKUP["play"])
926 elif state == self._player.STATE_STOP:
927 self._presenter.set_state(self._store.STORE_LOOKUP["play"])
929 _moduleLogger.info("Unhandled player state %s" % state)
931 @misc_utils.log_exception(_moduleLogger)
932 def _on_user_seek(self, widget, scroll, value):
933 self._player.seek(value / 100.0)
935 @misc_utils.log_exception(_moduleLogger)
936 def _on_player_update_seek(self):
937 self._seekbar.set_value(self._player.percent_elapsed * 100)
938 return True if not self._isDestroyed else False
940 @misc_utils.log_exception(_moduleLogger)
941 def _on_player_state_change(self, player, newState):
942 if self._active and self._player.state == self._player.STATE_PLAY:
944 assert self._updateSeek is None
945 self._updateSeek = go_utils.Timeout(self._updateSeek, once=False)
946 self._updateSeek.start(seconds=30)
949 self._updateSeek.cancel()
950 self._updateSeek = None
952 if not self._presenterNavigation.is_active():
953 self._set_context(newState)
955 @misc_utils.log_exception(_moduleLogger)
956 def _on_player_title_change(self, player, node):
957 if not self._active or node in [None, self._node]:
958 self._playerNode = player.node
960 self._playerNode = player.node
961 self.emit("jump-to", node)
962 self._window.destroy()
964 @misc_utils.log_exception(_moduleLogger)
965 def _on_player_error(self, player, err, debug):
966 _moduleLogger.error("%r - %r" % (err, debug))
968 @misc_utils.log_exception(_moduleLogger)
969 def _on_navigating(self, widget, navState):
970 if navState == "clicking":
971 if self._player.state == self._player.STATE_PLAY:
973 imageName = "pause_pressed"
975 imageName = "play_pressed"
976 elif self._player.state == self._player.STATE_PAUSE:
977 imageName = "play_pressed"
978 elif self._player.state == self._player.STATE_STOP:
979 imageName = "play_pressed"
981 _moduleLogger.info("Unhandled player state %s" % self._player.state)
982 elif navState == "down":
984 elif navState == "up":
985 if self._player.state == self._player.STATE_PLAY:
990 elif self._player.state == self._player.STATE_PAUSE:
992 elif self._player.state == self._player.STATE_STOP:
995 _moduleLogger.info("Unhandled player state %s" % self._player.state)
996 elif navState == "left":
998 elif navState == "right":
1001 self._presenter.set_state(self._store.STORE_LOOKUP[imageName])
1003 @misc_utils.log_exception(_moduleLogger)
1004 def _on_nav_action(self, widget, navState):
1005 self._set_context(self._player.state)
1007 if navState == "clicking":
1008 if self._player.state == self._player.STATE_PLAY:
1010 self._player.pause()
1012 self._player.set_piece_by_node(self._node)
1014 elif self._player.state == self._player.STATE_PAUSE:
1016 elif self._player.state == self._player.STATE_STOP:
1017 self._player.set_piece_by_node(self._node)
1020 _moduleLogger.info("Unhandled player state %s" % self._player.state)
1021 elif navState == "down":
1023 self._window.destroy()
1024 elif navState == "up":
1026 elif navState == "left":
1030 assert self._nextSearch is None
1031 self._nextSearch = stream_index.AsyncWalker(stream_index.get_next)
1032 self._nextSearch.start(self._node, self._on_next_node, self._on_node_search_error)
1033 elif navState == "right":
1037 assert self._nextSearch is None
1038 self._nextSearch = stream_index.AsyncWalker(stream_index.get_previous)
1039 self._nextSearch.start(self._node, self._on_next_node, self._on_node_search_error)
1041 @misc_utils.log_exception(_moduleLogger)
1042 def _on_next_node(self, node):
1043 self._nextSearch = None
1044 self.emit("jump-to", node)
1045 self._window.destroy()
1047 @misc_utils.log_exception(_moduleLogger)
1048 def _on_node_search_error(self, e):
1049 self._nextSearch = None
1050 self._errorBanner.push_message(str(e))
1053 gobject.type_register(ConferenceTalkWindow)