Switching over to the nav box plus a bug fix
[watersofshiloah] / src / windows.py
index dff9f9a..ba3f65d 100644 (file)
@@ -27,20 +27,34 @@ class BasicWindow(gobject.GObject):
                        gobject.TYPE_NONE,
                        (),
                ),
-               'fullscreen' : (
+               'home' : (
+                       gobject.SIGNAL_RUN_LAST,
+                       gobject.TYPE_NONE,
+                       (),
+               ),
+               'jump-to' : (
                        gobject.SIGNAL_RUN_LAST,
                        gobject.TYPE_NONE,
                        (gobject.TYPE_PYOBJECT, ),
                ),
+               'rotate' : (
+                       gobject.SIGNAL_RUN_LAST,
+                       gobject.TYPE_NONE,
+                       (gobject.TYPE_BOOLEAN, ),
+               ),
+               'fullscreen' : (
+                       gobject.SIGNAL_RUN_LAST,
+                       gobject.TYPE_NONE,
+                       (gobject.TYPE_BOOLEAN, ),
+               ),
        }
 
-       def __init__(self, player, store, index):
+       def __init__(self, player, store):
                gobject.GObject.__init__(self)
                self._isDestroyed = False
 
                self._player = player
                self._store = store
-               self._index = index
 
                self._clipboard = gtk.clipboard_get()
                self._windowInFullscreen = False
@@ -63,6 +77,9 @@ class BasicWindow(gobject.GObject):
        def window(self):
                return self._window
 
+       def show(self):
+               self._window.show_all()
+
        def save_settings(self, config, sectionName):
                config.add_section(sectionName)
                config.set(sectionName, "fullscreen", str(self._windowInFullscreen))
@@ -126,23 +143,39 @@ class BasicWindow(gobject.GObject):
                                self._clipboard.set_text(str(log))
                        return True
 
+       @misc_utils.log_exception(_moduleLogger)
+       def _on_home(self, *args):
+               self.emit("home")
+               self._window.destroy()
+
+       @misc_utils.log_exception(_moduleLogger)
+       def _on_jump(self, source, node):
+               _moduleLogger.error("Jump is not implemented")
+               self.emit("jump-to", node)
+               self._window.destroy()
+
+       @misc_utils.log_exception(_moduleLogger)
+       def _on_quit(self, *args):
+               self.emit("quit")
+               self._window.destroy()
+
 
 class SourceSelector(BasicWindow):
 
        def __init__(self, player, store, index):
+               BasicWindow.__init__(self, player, store)
                self._languages = []
-
-               BasicWindow.__init__(self, player, store, index)
+               self._index = index
 
                self._loadingBanner = banners.GenericBanner()
 
                self._radioButton = self._create_button("radio", "Radio")
-               self._radioButton.connect("clicked", self._on_source_selected, RadioWindow)
+               self._radioButton.connect("clicked", self._on_source_selected, RadioWindow, "radio")
                self._radioWrapper = gtk.VBox()
                self._radioWrapper.pack_start(self._radioButton, False, True)
 
                self._conferenceButton = self._create_button("conferences", "Conferences")
-               self._conferenceButton.connect("clicked", self._on_source_selected, ConferencesWindow)
+               self._conferenceButton.connect("clicked", self._on_source_selected, ConferencesWindow, "conferences")
                self._conferenceWrapper = gtk.VBox()
                self._conferenceWrapper.pack_start(self._conferenceButton, False, True)
 
@@ -163,14 +196,17 @@ class SourceSelector(BasicWindow):
                self._buttonLayout.pack_start(self._magazineWrapper, True, True)
                self._buttonLayout.pack_start(self._scriptureWrapper, True, True)
 
-               self._playcontrol = playcontrol.PlayControl(player, store)
+               self._playcontrol = playcontrol.NavControl(player, store)
 
                self._layout.pack_start(self._loadingBanner.toplevel, False, False)
                self._layout.pack_start(self._buttonLayout, True, True)
                self._layout.pack_start(self._playcontrol.toplevel, False, True)
 
                self._window.set_title(constants.__pretty_app_name__)
-               self._window.show_all()
+
+       def show(self):
+               BasicWindow.show(self)
+
                self._errorBanner.toplevel.hide()
                self._playcontrol.toplevel.hide()
 
@@ -188,11 +224,7 @@ class SourceSelector(BasicWindow):
 
        def _refresh(self):
                self._show_loading()
-               self._index.download(
-                       "get_languages",
-                       self._on_languages,
-                       self._on_error,
-               )
+               self._index.get_languages(self._on_languages, self._on_error)
 
        def _create_button(self, icon, message):
                image = self._store.get_image_from_store(self._store.STORE_LOOKUP[icon])
@@ -216,14 +248,17 @@ class SourceSelector(BasicWindow):
        @misc_utils.log_exception(_moduleLogger)
        def _on_error(self, exception):
                self._hide_loading()
-               self._errorBanner.push_message(exception)
+               self._errorBanner.push_message(str(exception))
 
        @misc_utils.log_exception(_moduleLogger)
-       def _on_source_selected(self, widget, Source):
-               sourceWindow = Source(self._player, self._store, self._index, self._languages[0]["id"])
+       def _on_source_selected(self, widget, Source, nodeName):
+               node = self._index.get_source(nodeName, self._languages[0]["id"])
+               sourceWindow = Source(self._player, self._store, node)
                sourceWindow.window.set_modal(True)
                sourceWindow.window.set_transient_for(self._window)
                sourceWindow.window.set_default_size(*self._window.get_size())
+               sourceWindow.connect("quit", self._on_quit)
+               sourceWindow.show()
 
 
 gobject.type_register(SourceSelector)
@@ -231,8 +266,13 @@ gobject.type_register(SourceSelector)
 
 class RadioWindow(BasicWindow):
 
-       def __init__(self, player, store, index, languageId):
-               BasicWindow.__init__(self, player, store, index)
+       def __init__(self, player, store, node):
+               BasicWindow.__init__(self, player, store)
+               self._node = node
+               self._childNode = None
+
+               self._player.connect("state-change", self._on_player_state_change)
+               self._player.connect("title-change", self._on_player_title_change)
 
                self._loadingBanner = banners.GenericBanner()
 
@@ -241,7 +281,7 @@ class RadioWindow(BasicWindow):
                self._headerNavigation = presenter.NavigationBox()
                self._headerNavigation.toplevel.add(self._header)
                self._headerNavigation.connect("action", self._on_nav_action)
-
+               self._headerNavigation.connect("navigating", self._on_navigating)
 
                self._programmingModel = gtk.ListStore(
                        gobject.TYPE_STRING,
@@ -269,10 +309,15 @@ class RadioWindow(BasicWindow):
                self._treeScroller.add(self._treeView)
                self._treeScroller.set_policy(gtk.POLICY_NEVER, gtk.POLICY_AUTOMATIC)
 
-               self._presenter = presenter.StreamMiniPresenter(self._player, self._store)
+               self._presenter = presenter.StreamMiniPresenter(self._store)
+               if self._player.state == "play":
+                       self._presenter.set_state(self._store.STORE_LOOKUP["play"])
+               else:
+                       self._presenter.set_state(self._store.STORE_LOOKUP["pause"])
                self._presenterNavigation = presenter.NavigationBox()
                self._presenterNavigation.toplevel.add(self._presenter.toplevel)
                self._presenterNavigation.connect("action", self._on_nav_action)
+               self._presenterNavigation.connect("navigating", self._on_navigating)
 
                self._radioLayout = gtk.VBox(False)
                self._radioLayout.pack_start(self._headerNavigation.toplevel, False, False)
@@ -283,11 +328,14 @@ class RadioWindow(BasicWindow):
                self._layout.pack_start(self._radioLayout, True, True)
 
                self._window.set_title("Radio")
-               self._window.show_all()
+               self._dateShown = datetime.datetime.now()
+
+       def show(self):
+               BasicWindow.show(self)
+
                self._errorBanner.toplevel.hide()
                self._loadingBanner.toplevel.hide()
 
-               self._dateShown = datetime.datetime.now()
                self._refresh()
 
        def _show_loading(self):
@@ -301,8 +349,7 @@ class RadioWindow(BasicWindow):
        def _refresh(self):
                self._show_loading()
                self._programmingModel.clear()
-               self._index.download(
-                       "get_radio_channels",
+               self._node.get_children(
                        self._on_channels,
                        self._on_load_error,
                )
@@ -320,9 +367,53 @@ class RadioWindow(BasicWindow):
                        return i
 
        @misc_utils.log_exception(_moduleLogger)
+       def _on_player_state_change(self, player, newState):
+               if self._headerNavigation.is_active() or self._presenterNavigation.is_active():
+                       return
+
+               if newState == "play":
+                       self._presenter.set_state(self._store.STORE_LOOKUP["play"])
+               elif newState == "pause":
+                       self._presenter.set_state(self._store.STORE_LOOKUP["pause"])
+               else:
+                       _moduleLogger.info("Unhandled player state %s" % newState)
+                       self._presenter.set_state(self._store.STORE_LOOKUP["pause"])
+
+       @misc_utils.log_exception(_moduleLogger)
+       def _on_player_title_change(self, player, newState):
+               _moduleLogger.info("Player title magically changed to %s" % player.title)
+               self._destroy()
+
+       @misc_utils.log_exception(_moduleLogger)
+       def _on_navigating(self, widget, navState):
+               if navState == "clicking":
+                       if self._player.state == "play":
+                               imageName = "pause"
+                       else:
+                               imageName = "play"
+               elif navState == "down":
+                       imageName = "home"
+               elif navState == "up":
+                       imageName = "play"
+               elif navState == "left":
+                       imageName = "play"
+               elif navState == "right":
+                       imageName = "play"
+
+               self._presenter.set_state(self._store.STORE_LOOKUP[imageName])
+
+       @misc_utils.log_exception(_moduleLogger)
        def _on_nav_action(self, widget, navState):
+               if self._player.state == "play":
+                       self._presenter.set_state(self._store.STORE_LOOKUP["play"])
+               else:
+                       self._presenter.set_state(self._store.STORE_LOOKUP["pause"])
+
                if navState == "clicking":
-                       pass
+                       if self._player.state == "play":
+                               self._player.pause()
+                       else:
+                               self._player.play()
                elif navState == "down":
                        self.window.destroy()
                elif navState == "up":
@@ -340,16 +431,14 @@ class RadioWindow(BasicWindow):
                        _moduleLogger.info("Download complete but window destroyed")
                        return
 
-               channels = list(channels)
+               channels = channels
                if 1 < len(channels):
                        _moduleLogger.warning("More channels now available!")
-               channel = channels[0]
-               self._index.download(
-                       "get_radio_channel_programming",
+               self._childNode = channels[0]
+               self._childNode.get_programming(
+                       self._dateShown,
                        self._on_channel,
                        self._on_load_error,
-                       channel["id"],
-                       self._dateShown,
                )
 
        @misc_utils.log_exception(_moduleLogger)
@@ -370,7 +459,7 @@ class RadioWindow(BasicWindow):
        @misc_utils.log_exception(_moduleLogger)
        def _on_load_error(self, exception):
                self._hide_loading()
-               self._errorBanner.push_message(exception)
+               self._errorBanner.push_message(str(exception))
 
        @misc_utils.log_exception(_moduleLogger)
        def _on_row_changed(self, selection):
@@ -389,8 +478,9 @@ gobject.type_register(RadioWindow)
 
 class ListWindow(BasicWindow):
 
-       def __init__(self, player, store, index):
-               BasicWindow.__init__(self, player, store, index)
+       def __init__(self, player, store, node):
+               BasicWindow.__init__(self, player, store)
+               self._node = node
 
                self._loadingBanner = banners.GenericBanner()
 
@@ -410,7 +500,9 @@ class ListWindow(BasicWindow):
                self._treeScroller.add(self._treeView)
                self._treeScroller.set_policy(gtk.POLICY_NEVER, gtk.POLICY_AUTOMATIC)
 
-               self._playcontrol = playcontrol.PlayControl(self._player, self._store)
+               self._playcontrol = playcontrol.NavControl(self._player, self._store)
+               self._playcontrol.connect("home", self._on_home)
+               self._playcontrol.connect("jump-to", self._on_jump)
 
                self._contentLayout = gtk.VBox(False)
                self._contentLayout.pack_start(self._treeScroller, True, True)
@@ -419,7 +511,9 @@ class ListWindow(BasicWindow):
                self._layout.pack_start(self._loadingBanner.toplevel, False, False)
                self._layout.pack_start(self._contentLayout, True, True)
 
-               self._window.show_all()
+       def show(self):
+               BasicWindow.show(self)
+
                self._errorBanner.toplevel.hide()
                self._loadingBanner.toplevel.hide()
 
@@ -457,15 +551,13 @@ class ListWindow(BasicWindow):
 
 class ConferencesWindow(ListWindow):
 
-       def __init__(self, player, store, index, languageId):
-               self._languageId = languageId
-
-               ListWindow.__init__(self, player, store, index)
+       def __init__(self, player, store, node):
+               ListWindow.__init__(self, player, store, node)
                self._window.set_title("Conferences")
 
        @classmethod
        def _get_columns(cls):
-               yield gobject.TYPE_STRING, None
+               yield gobject.TYPE_PYOBJECT, None
 
                textrenderer = gtk.CellRendererText()
                column = gtk.TreeViewColumn("Date")
@@ -485,11 +577,9 @@ class ConferencesWindow(ListWindow):
 
        def _refresh(self):
                ListWindow._refresh(self)
-               self._index.download(
-                       "get_conferences",
+               self._node.get_children(
                        self._on_conferences,
                        self._on_error,
-                       self._languageId,
                )
 
        @misc_utils.log_exception(_moduleLogger)
@@ -499,8 +589,9 @@ class ConferencesWindow(ListWindow):
                        return
 
                self._hide_loading()
-               for program in programs:
-                       row = program["id"], program["title"], program["full_title"]
+               for programNode in programs:
+                       program = programNode.get_properties()
+                       row = programNode, program["title"], program["full_title"]
                        self._model.append(row)
 
                path = (self._get_current_row(), )
@@ -510,17 +601,20 @@ class ConferencesWindow(ListWindow):
        @misc_utils.log_exception(_moduleLogger)
        def _on_error(self, exception):
                self._hide_loading()
-               self._errorBanner.push_message(exception)
+               self._errorBanner.push_message(str(exception))
 
        @misc_utils.log_exception(_moduleLogger)
        def _on_row_activated(self, view, path, column):
                itr = self._model.get_iter(path)
-               conferenceId = self._model.get_value(itr, 0)
+               node = self._model.get_value(itr, 0)
 
-               sessionsWindow = ConferenceSessionsWindow(self._player, self._store, self._index, conferenceId)
+               sessionsWindow = ConferenceSessionsWindow(self._player, self._store, node)
                sessionsWindow.window.set_modal(True)
                sessionsWindow.window.set_transient_for(self._window)
                sessionsWindow.window.set_default_size(*self._window.get_size())
+               sessionsWindow.connect("quit", self._on_quit)
+               sessionsWindow.connect("home", self._on_home)
+               sessionsWindow.show()
 
 
 gobject.type_register(ConferencesWindow)
@@ -528,15 +622,13 @@ gobject.type_register(ConferencesWindow)
 
 class ConferenceSessionsWindow(ListWindow):
 
-       def __init__(self, player, store, index, conferenceId):
-               self._conferenceId = conferenceId
-
-               ListWindow.__init__(self, player, store, index)
+       def __init__(self, player, store, node):
+               ListWindow.__init__(self, player, store, node)
                self._window.set_title("Sessions")
 
        @classmethod
        def _get_columns(cls):
-               yield gobject.TYPE_STRING, None
+               yield gobject.TYPE_PYOBJECT, None
 
                textrenderer = gtk.CellRendererText()
                column = gtk.TreeViewColumn("Session")
@@ -550,11 +642,9 @@ class ConferenceSessionsWindow(ListWindow):
 
        def _refresh(self):
                ListWindow._refresh(self)
-               self._index.download(
-                       "get_conference_sessions",
+               self._node.get_children(
                        self._on_conference_sessions,
                        self._on_error,
-                       self._conferenceId,
                )
 
        @misc_utils.log_exception(_moduleLogger)
@@ -564,8 +654,9 @@ class ConferenceSessionsWindow(ListWindow):
                        return
 
                self._hide_loading()
-               for program in programs:
-                       row = program["id"], program["title"]
+               for programNode in programs:
+                       program = programNode.get_properties()
+                       row = programNode, program["title"]
                        self._model.append(row)
 
                path = (self._get_current_row(), )
@@ -575,17 +666,20 @@ class ConferenceSessionsWindow(ListWindow):
        @misc_utils.log_exception(_moduleLogger)
        def _on_error(self, exception):
                self._hide_loading()
-               self._errorBanner.push_message(exception)
+               self._errorBanner.push_message(str(exception))
 
        @misc_utils.log_exception(_moduleLogger)
        def _on_row_activated(self, view, path, column):
                itr = self._model.get_iter(path)
-               sessionId = self._model.get_value(itr, 0)
+               node = self._model.get_value(itr, 0)
 
-               sessionsWindow = ConferenceTalksWindow(self._player, self._store, self._index, sessionId)
+               sessionsWindow = ConferenceTalksWindow(self._player, self._store, node)
                sessionsWindow.window.set_modal(True)
                sessionsWindow.window.set_transient_for(self._window)
                sessionsWindow.window.set_default_size(*self._window.get_size())
+               sessionsWindow.connect("quit", self._on_quit)
+               sessionsWindow.connect("home", self._on_home)
+               sessionsWindow.show()
 
 
 gobject.type_register(ConferenceSessionsWindow)
@@ -593,15 +687,13 @@ gobject.type_register(ConferenceSessionsWindow)
 
 class ConferenceTalksWindow(ListWindow):
 
-       def __init__(self, player, store, index, sessionId):
-               self._sessionId = sessionId
-
-               ListWindow.__init__(self, player, store, index)
+       def __init__(self, player, store, node):
+               ListWindow.__init__(self, player, store, node)
                self._window.set_title("Talks")
 
        @classmethod
        def _get_columns(cls):
-               yield gobject.TYPE_STRING, None
+               yield gobject.TYPE_PYOBJECT, None
 
                textrenderer = gtk.CellRendererText()
                column = gtk.TreeViewColumn("Talk")
@@ -615,11 +707,9 @@ class ConferenceTalksWindow(ListWindow):
 
        def _refresh(self):
                ListWindow._refresh(self)
-               self._index.download(
-                       "get_conference_talks",
+               self._node.get_children(
                        self._on_conference_talks,
                        self._on_error,
-                       self._sessionId,
                )
 
        @misc_utils.log_exception(_moduleLogger)
@@ -629,8 +719,9 @@ class ConferenceTalksWindow(ListWindow):
                        return
 
                self._hide_loading()
-               for program in programs:
-                       row = program["id"], "%s\n%s" % (program["title"], program["speaker"])
+               for programNode in programs:
+                       program = programNode.get_properties()
+                       row = programNode, "%s\n%s" % (program["title"], program["speaker"])
                        self._model.append(row)
 
                path = (self._get_current_row(), )
@@ -640,11 +731,130 @@ class ConferenceTalksWindow(ListWindow):
        @misc_utils.log_exception(_moduleLogger)
        def _on_error(self, exception):
                self._hide_loading()
-               self._errorBanner.push_message(exception)
+               self._errorBanner.push_message(str(exception))
 
        @misc_utils.log_exception(_moduleLogger)
        def _on_row_activated(self, view, path, column):
-               raise NotImplementedError("")
+               itr = self._model.get_iter(path)
+               node = self._model.get_value(itr, 0)
+
+               sessionsWindow = ConferenceTalkWindow(self._player, self._store, node)
+               sessionsWindow.window.set_modal(True)
+               sessionsWindow.window.set_transient_for(self._window)
+               sessionsWindow.window.set_default_size(*self._window.get_size())
+               sessionsWindow.connect("quit", self._on_quit)
+               sessionsWindow.connect("home", self._on_home)
+               sessionsWindow.show()
 
 
 gobject.type_register(ConferenceTalksWindow)
+
+
+class ConferenceTalkWindow(BasicWindow):
+
+       def __init__(self, player, store, node):
+               BasicWindow.__init__(self, player, store)
+
+               self._player.connect("state-change", self._on_player_state_change)
+               self._player.connect("title-change", self._on_player_title_change)
+
+               self._loadingBanner = banners.GenericBanner()
+
+               self._presenter = presenter.StreamPresenter(self._store)
+               self._presenterNavigation = presenter.NavigationBox()
+               self._presenterNavigation.toplevel.add(self._presenter.toplevel)
+               self._presenterNavigation.connect("action", self._on_nav_action)
+               self._presenterNavigation.connect("navigating", self._on_navigating)
+
+               self._layout.pack_start(self._loadingBanner.toplevel, False, False)
+               self._layout.pack_start(self._presenterNavigation.toplevel, True, True)
+
+               self._window.set_title("Talk")
+
+       def show(self):
+               BasicWindow.show(self)
+               self._window.show_all()
+               self._errorBanner.toplevel.hide()
+               self._loadingBanner.toplevel.hide()
+
+               self._presenter.set_context(
+                       self._store.STORE_LOOKUP["conference_background"],
+                       self._player.title,
+                       self._player.subtitle,
+               )
+               if self._player.state == "play":
+                       self._presenter.set_state(self._store.STORE_LOOKUP["play"])
+               else:
+                       self._presenter.set_state(self._store.STORE_LOOKUP["pause"])
+
+       def _show_loading(self):
+               animationPath = self._store.STORE_LOOKUP["loading"]
+               animation = self._store.get_pixbuf_animation_from_store(animationPath)
+               self._loadingBanner.show(animation, "Loading...")
+
+       def _hide_loading(self):
+               self._loadingBanner.hide()
+
+       @misc_utils.log_exception(_moduleLogger)
+       def _on_player_state_change(self, player, newState):
+               if self._presenterNavigation.is_active():
+                       return
+
+               if newState == "play":
+                       self._presenter.set_state(self._store.STORE_LOOKUP["play"])
+               elif newState == "pause":
+                       self._presenter.set_state(self._store.STORE_LOOKUP["pause"])
+               else:
+                       _moduleLogger.info("Unhandled player state %s" % newState)
+                       self._presenter.set_state(self._store.STORE_LOOKUP["pause"])
+
+       @misc_utils.log_exception(_moduleLogger)
+       def _on_player_title_change(self, player, newState):
+               self._presenter.set_context(
+                       self._store.STORE_LOOKUP["conference_background"],
+                       self._player.title,
+                       self._player.subtitle,
+               )
+
+       @misc_utils.log_exception(_moduleLogger)
+       def _on_navigating(self, widget, navState):
+               if navState == "clicking":
+                       if self._player.state == "play":
+                               imageName = "pause"
+                       else:
+                               imageName = "play"
+               elif navState == "down":
+                       imageName = "home"
+               elif navState == "up":
+                       imageName = "play"
+               elif navState == "left":
+                       imageName = "next"
+               elif navState == "right":
+                       imageName = "prev"
+
+               self._presenter.set_state(self._store.STORE_LOOKUP[imageName])
+
+       @misc_utils.log_exception(_moduleLogger)
+       def _on_nav_action(self, widget, navState):
+               if self._player.state == "play":
+                       self._presenter.set_state(self._store.STORE_LOOKUP["play"])
+               else:
+                       self._presenter.set_state(self._store.STORE_LOOKUP["pause"])
+
+               if navState == "clicking":
+                       if self._player.state == "play":
+                               self._player.pause()
+                       else:
+                               self._player.play()
+               elif navState == "down":
+                       self.emit("home")
+                       self._window.destroy()
+               elif navState == "up":
+                       pass
+               elif navState == "left":
+                       self._player.next()
+               elif navState == "right":
+                       self._player.back()
+
+
+gobject.type_register(ConferenceTalkWindow)