← Back to team overview

openlp-core team mailing list archive

[Merge] lp:~crichter/openlp/media_rewrite into lp:openlp

 

rimach has proposed merging lp:~crichter/openlp/media_rewrite into lp:openlp.

Requested reviews:
  Andreas Preikschat (googol)
  Raoul Snyman (raoul-snyman)
  Tim Bentley (trb143)
  Jonathan Corwin (j-corwin)
Related bugs:
  Bug #813995 in OpenLP: "opengl rendering"
  https://bugs.launchpad.net/openlp/+bug/813995

For more details, see:
https://code.launchpad.net/~crichter/openlp/media_rewrite/+merge/84011

Rewrite of the multimedia stuff.
- add separated multimedia code in own subdirectory in openlp/ui


-- 
https://code.launchpad.net/~crichter/openlp/media_rewrite/+merge/84011
Your team OpenLP Core is subscribed to branch lp:openlp.
=== modified file 'openlp/core/lib/htmlbuilder.py'
--- openlp/core/lib/htmlbuilder.py	2011-11-24 21:25:38 +0000
+++ openlp/core/lib/htmlbuilder.py	2011-11-30 20:32:27 +0000
@@ -53,8 +53,8 @@
     position: absolute;
     left: 0px;
     top: 0px;
-    width: %spx;
-    height: %spx;
+    width: 100%%;
+    height: 100%%;
 }
 #black {
     z-index: 8;
@@ -67,12 +67,6 @@
 #image {
     z-index: 2;
 }
-#video1 {
-    z-index: 3;
-}
-#video2 {
-    z-index: 3;
-}
 %s
 #footer {
     position: absolute;
@@ -90,90 +84,9 @@
 </style>
 <script>
     var timer = null;
-    var video_timer = null;
-    var current_video = '1';
     var transition = %s;
-
-    function show_video(state, path, volume, loop){
-        // Note, the preferred method for looping would be to use the
-        // video tag loop attribute.
-        // But QtWebKit doesn't support this. Neither does it support the
-        // onended event, hence the setInterval()
-        // In addition, setting the currentTime attribute to zero to restart
-        // the video raises an INDEX_SIZE_ERROR: DOM Exception 1
-        // To complicate it further, sometimes vid.currentTime stops
-        // slightly short of vid.duration and vid.ended is intermittent!
-        //
-        // Note, currently the background may go black between loops. Not
-        // desirable. Need to investigate using two <video>'s, and hiding/
-        // preloading one, and toggle between the two when looping.
-
-        if(current_video=='1'){
-            var vid = document.getElementById('video1');
-            var vid2 = document.getElementById('video2');
-        } else {
-            var vid = document.getElementById('video2');
-            var vid2 = document.getElementById('video1');
-        }
-        if(volume != null){
-            vid.volume = volume;
-            vid2.volume = volume;
-        }
-        switch(state){
-            case 'init':
-                vid.src = 'file:///' + path;
-                vid2.src = 'file:///' + path;
-                if(loop == null) loop = false;
-                vid.looping = loop;
-                vid2.looping = loop;
-                vid.load();
-                break;
-            case 'load':
-                vid2.style.visibility = 'hidden';
-                vid2.load();
-                break;
-            case 'play':
-                vid.play();
-                vid.style.visibility = 'visible';
-                if(vid.looping){
-                    video_timer = setInterval(
-                        function() {
-                            show_video('poll');
-                        }, 200);
-                }
-                break;
-            case 'pause':
-                if(video_timer!=null){
-                    clearInterval(video_timer);
-                    video_timer = null;
-                }
-                vid.pause();
-                break;
-            case 'stop':
-                show_video('pause');
-                vid.style.visibility = 'hidden';
-                break;
-            case 'poll':
-                if(vid.ended||vid.currentTime+0.2>vid.duration)
-                    show_video('swap');
-                break;
-            case 'swap':
-                show_video('pause');
-                if(current_video=='1')
-                    current_video = '2';
-                else
-                    current_video = '1';
-                show_video('play');
-                show_video('load');
-                break;
-            case 'close':
-                show_video('stop');
-                vid.src = '';
-                vid2.src = '';
-                break;
-        }
-    }
     %s
+
     function show_image(src){
         var img = document.getElementById('image');
         img.src = src;
@@ -186,18 +99,14 @@
     function show_blank(state){
         var black = 'none';
         var lyrics = '';
-        var pause = false;
         switch(state){
             case 'theme':
                 lyrics = 'hidden';
-                pause = true;
                 break;
             case 'black':
                 black = 'block';
-                pause = true;
                 break;
             case 'desktop':
-                pause = true;
                 break;
         }
         document.getElementById('black').style.display = black;
@@ -210,13 +119,6 @@
         if(shadow!=null)
             shadow.style.visibility = lyrics;
         document.getElementById('footer').style.visibility = lyrics;
-        var vid = document.getElementById('video');
-        if(vid.src != ''){
-            if(pause)
-                vid.pause();
-            else
-                vid.play();
-        }
     }
 
     function show_footer(footertext){
@@ -277,10 +179,6 @@
 <body>
 <img id="bgimage" class="size" %s />
 <img id="image" class="size" %s />
-<video id="video1" class="size" style="visibility:hidden" autobuffer preload>
-</video>
-<video id="video2" class="size" style="visibility:hidden" autobuffer preload>
-</video>
 %s
 %s
 <div id="footer" class="footer"></div>
@@ -336,7 +234,6 @@
             js_additions += plugin.getDisplayJavaScript()
             html_additions += plugin.getDisplayHtml()
     html = HTMLSRC % (build_background_css(item, width, height),
-        width, height,
         css_additions,
         build_footer_css(item, height),
         build_lyrics_css(item, webkitvers),
@@ -609,4 +506,3 @@
         item.footer.width(), theme.font_footer_name,
         theme.font_footer_size, theme.font_footer_color)
     return lyrics_html
-

=== added file 'openlp/core/lib/mediaplayer.py'
--- openlp/core/lib/mediaplayer.py	1970-01-01 00:00:00 +0000
+++ openlp/core/lib/mediaplayer.py	2011-11-30 20:32:27 +0000
@@ -0,0 +1,137 @@
+# -*- coding: utf-8 -*-
+# vim: autoindent shiftwidth=4 expandtab textwidth=80 tabstop=4 softtabstop=4
+
+###############################################################################
+# OpenLP - Open Source Lyrics Projection                                      #
+# --------------------------------------------------------------------------- #
+# Copyright (c) 2008-2011 Raoul Snyman                                        #
+# Portions copyright (c) 2008-2011 Tim Bentley, Gerald Britton, Jonathan      #
+# Corwin, Michael Gorven, Scott Guerrieri, Matthias Hub, Meinert Jordan,      #
+# Armin Köhler, Joshua Miller, Stevan Pettit, Andreas Preikschat, Mattias     #
+# Põldaru, Christian Richter, Philip Ridout, Simon Scudder, Jeffrey Smith,    #
+# Maikel Stuivenberg, Martin Thompson, Jon Tibble, Frode Woldsund             #
+# --------------------------------------------------------------------------- #
+# This program is free software; you can redistribute it and/or modify it     #
+# under the terms of the GNU General Public License as published by the Free  #
+# Software Foundation; version 2 of the License.                              #
+#                                                                             #
+# This program is distributed in the hope that it will be useful, but WITHOUT #
+# ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or       #
+# FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for    #
+# more details.                                                               #
+#                                                                             #
+# You should have received a copy of the GNU General Public License along     #
+# with this program; if not, write to the Free Software Foundation, Inc., 59  #
+# Temple Place, Suite 330, Boston, MA 02111-1307 USA                          #
+###############################################################################
+
+from openlp.core.ui.media import MediaState
+
+class MediaPlayer(object):
+    """
+    This is the base class media Player class to provide OpenLP with a pluggable media display 
+    framework.
+    """
+
+    def __init__(self, parent, name=u'media_player'):
+        self.parent = parent
+        self.name = name
+        self.available = self.check_available()
+        self.isActive = False
+        self.canBackground = False
+        self.canFolder = False
+        self.state = MediaState.Off
+        self.hasOwnWidget = False
+        self.audio_extensions_list = []
+        self.video_extensions_list = []
+
+    def check_available(self):
+        """
+        Player is available on this machine
+        """
+        return False
+
+    def setup(self, display):
+        """
+        Create the related widgets for the current display
+        """
+        pass
+
+    def load(self, display):
+        """
+        Load a new media file and check if it is valid
+        """
+        return True
+
+    def resize(self, display):
+        """
+        If the main display size or position is changed, the media widgets 
+        should also resized
+        """
+        pass
+
+    def play(self, display):
+        """
+        Starts playing of current Media File
+        """
+        pass
+
+    def pause(self, display):
+        """
+        Pause of current Media File
+        """
+        pass
+
+    def stop(self, display):
+        """
+        Stop playing of current Media File
+        """
+        pass
+
+    def volume(self, display, vol):
+        """
+        Change volume of current Media File
+        """
+        pass
+
+    def seek(self, display, seekVal):
+        """
+        Change playing position of current Media File
+        """
+        pass
+
+    def reset(self, display):
+        """
+        Remove the current loaded video
+        """
+        pass
+
+    def set_visible(self, display, status):
+        """
+        Show/Hide the media widgets
+        """
+        pass
+
+    def update_ui(self, display):
+        """
+        Do some ui related stuff (e.g. update the seek slider)
+        """
+        pass
+
+    def get_media_display_css(self):
+        """
+        Add css style sheets to htmlbuilder
+        """
+        return u''
+
+    def get_media_display_javascript(self):
+        """
+        Add javascript functions to htmlbuilder
+        """
+        return u''
+
+    def get_media_display_html(self):
+        """
+        Add html code to htmlbuilder
+        """
+        return u''

=== modified file 'openlp/core/lib/plugin.py'
--- openlp/core/lib/plugin.py	2011-10-17 18:01:07 +0000
+++ openlp/core/lib/plugin.py	2011-11-30 20:32:27 +0000
@@ -168,6 +168,7 @@
         self.mediadock = plugin_helpers[u'toolbox']
         self.pluginManager = plugin_helpers[u'pluginmanager']
         self.formparent = plugin_helpers[u'formparent']
+        self.mediaController = plugin_helpers[u'mediacontroller']
         QtCore.QObject.connect(Receiver.get_receiver(),
             QtCore.SIGNAL(u'%s_add_service_item' % self.name),
             self.processAddServiceEvent)
@@ -395,4 +396,3 @@
         Add html code to htmlbuilder.
         """
         return u''
-

=== modified file 'openlp/core/lib/renderer.py'
--- openlp/core/lib/renderer.py	2011-11-02 20:09:06 +0000
+++ openlp/core/lib/renderer.py	2011-11-30 20:32:27 +0000
@@ -76,7 +76,7 @@
         self.theme_data = None
         self.bg_frame = None
         self.force_page = False
-        self.display = MainDisplay(None, self.imageManager, False)
+        self.display = MainDisplay(None, self.imageManager, False, self)
         self.display.setup()
 
     def update_display(self):
@@ -87,7 +87,7 @@
         self._calculate_default()
         if self.display:
             self.display.close()
-        self.display = MainDisplay(None, self.imageManager, False)
+        self.display = MainDisplay(None, self.imageManager, False, self)
         self.display.setup()
         self.bg_frame = None
         self.theme_data = None

=== modified file 'openlp/core/ui/__init__.py'
--- openlp/core/ui/__init__.py	2011-10-29 19:13:11 +0000
+++ openlp/core/ui/__init__.py	2011-11-30 20:32:27 +0000
@@ -77,10 +77,10 @@
 from filerenameform import FileRenameForm
 from starttimeform import StartTimeForm
 from screen import ScreenList
-from maindisplay import MainDisplay
+from maindisplay import MainDisplay, Display
 from servicenoteform import ServiceNoteForm
 from serviceitemeditform import ServiceItemEditForm
-from slidecontroller import SlideController
+from slidecontroller import SlideController, Controller
 from splashscreen import SplashScreen
 from generaltab import GeneralTab
 from themestab import ThemesTab

=== modified file 'openlp/core/ui/maindisplay.py'
--- openlp/core/ui/maindisplay.py	2011-11-01 06:09:21 +0000
+++ openlp/core/ui/maindisplay.py	2011-11-30 20:32:27 +0000
@@ -31,7 +31,7 @@
 import logging
 import os
 
-from PyQt4 import QtCore, QtGui, QtWebKit
+from PyQt4 import QtCore, QtGui, QtWebKit, QtOpenGL
 from PyQt4.phonon import Phonon
 
 from openlp.core.lib import Receiver, build_html, ServiceItem, image_to_byte, \
@@ -44,11 +44,13 @@
 #http://www.steveheffernan.com/html5-video-player/demo-video-player.html
 #http://html5demos.com/two-videos
 
-class MainDisplay(QtGui.QGraphicsView):
-    """
-    This is the display screen.
-    """
-    def __init__(self, parent, imageManager, live):
+class Display(QtGui.QGraphicsView):
+    """
+    This is a general display screen class. Here the general display settings 
+   will done. It will be used as specialized classes by Main Display and
+   Preview display.
+    """
+    def __init__(self, parent, live, controller):
         if live:
             QtGui.QGraphicsView.__init__(self)
             # Overwrite the parent() method.
@@ -56,12 +58,60 @@
         else:
             QtGui.QGraphicsView.__init__(self, parent)
         self.isLive = live
+        self.controller = controller
+        self.screen = {}
+        self.plugins = PluginManager.get_instance().plugins
+        self.setViewport(QtOpenGL.QGLWidget())
+
+    def setup(self):
+        """
+        Set up and build the screen base
+        """
+        log.debug(u'Start Display base setup (live = %s)' % self.isLive)        
+        self.setGeometry(self.screen[u'size'])
+        log.debug(u'Setup webView')
+        self.webView = QtWebKit.QWebView(self)
+        self.webView.setGeometry(0, 0,
+            self.screen[u'size'].width(), self.screen[u'size'].height())
+        self.webView.settings().setAttribute(
+            QtWebKit.QWebSettings.PluginsEnabled, True)
+        self.page = self.webView.page()
+        self.frame = self.page.mainFrame()
+        if self.isLive and log.getEffectiveLevel() == logging.DEBUG:
+            self.webView.settings().setAttribute(
+                QtWebKit.QWebSettings.DeveloperExtrasEnabled, True)
+        QtCore.QObject.connect(self.webView,
+            QtCore.SIGNAL(u'loadFinished(bool)'), self.isWebLoaded)
+        self.setVerticalScrollBarPolicy(QtCore.Qt.ScrollBarAlwaysOff)
+        self.setHorizontalScrollBarPolicy(QtCore.Qt.ScrollBarAlwaysOff)
+        self.frame.setScrollBarPolicy(QtCore.Qt.Vertical,
+            QtCore.Qt.ScrollBarAlwaysOff)
+        self.frame.setScrollBarPolicy(QtCore.Qt.Horizontal,
+            QtCore.Qt.ScrollBarAlwaysOff)
+
+    def resizeEvent(self, ev):
+        self.webView.setGeometry(0, 0,
+            self.width(), self.height())
+
+    def isWebLoaded(self):
+        """
+        Called by webView event to show display is fully loaded
+        """
+        log.debug(u'Webloaded')
+        self.webLoaded = True
+
+
+class MainDisplay(Display):
+    """
+    This is the display screen as a specialized class from the Display class
+    """
+    def __init__(self, parent, imageManager, live, controller):
+        Display.__init__(self, parent, live, controller)
         self.imageManager = imageManager
         self.screens = ScreenList.get_instance()
         self.plugins = PluginManager.get_instance().plugins
         self.rebuildCSS = False
         self.hideMode = None
-        self.videoHide = False
         self.override = {}
         self.retranslateUi()
         self.mediaObject = None
@@ -81,9 +131,6 @@
             QtCore.QObject.connect(Receiver.get_receiver(),
                 QtCore.SIGNAL(u'live_display_show'), self.showDisplay)
             QtCore.QObject.connect(Receiver.get_receiver(),
-                QtCore.SIGNAL(u'openlp_phonon_creation'),
-                self.createMediaObject)
-            QtCore.QObject.connect(Receiver.get_receiver(),
                 QtCore.SIGNAL(u'update_display_css'), self.cssChanged)
             QtCore.QObject.connect(Receiver.get_receiver(),
                 QtCore.SIGNAL(u'config_updated'), self.configChanged)
@@ -115,36 +162,9 @@
         Set up and build the output screen
         """
         log.debug(u'Start MainDisplay setup (live = %s)' % self.isLive)
-        self.usePhonon = QtCore.QSettings().value(
-            u'media/use phonon', QtCore.QVariant(True)).toBool()
-        self.phononActive = False
         self.screen = self.screens.current
         self.setVisible(False)
-        self.setGeometry(self.screen[u'size'])
-        self.videoWidget = Phonon.VideoWidget(self)
-        self.videoWidget.setVisible(False)
-        self.videoWidget.setGeometry(QtCore.QRect(0, 0,
-            self.screen[u'size'].width(), self.screen[u'size'].height()))
-        if self.isLive:
-            if not self.firstTime:
-                self.createMediaObject()
-        log.debug(u'Setup webView')
-        self.webView = QtWebKit.QWebView(self)
-        self.webView.setGeometry(0, 0,
-            self.screen[u'size'].width(), self.screen[u'size'].height())
-        self.page = self.webView.page()
-        self.frame = self.page.mainFrame()
-        if self.isLive and log.getEffectiveLevel() == logging.DEBUG:
-            self.webView.settings().setAttribute(
-                QtWebKit.QWebSettings.DeveloperExtrasEnabled, True)
-        QtCore.QObject.connect(self.webView,
-            QtCore.SIGNAL(u'loadFinished(bool)'), self.isWebLoaded)
-        self.setVerticalScrollBarPolicy(QtCore.Qt.ScrollBarAlwaysOff)
-        self.setHorizontalScrollBarPolicy(QtCore.Qt.ScrollBarAlwaysOff)
-        self.frame.setScrollBarPolicy(QtCore.Qt.Vertical,
-            QtCore.Qt.ScrollBarAlwaysOff)
-        self.frame.setScrollBarPolicy(QtCore.Qt.Horizontal,
-            QtCore.Qt.ScrollBarAlwaysOff)
+        Display.setup(self)
         if self.isLive:
             # Build the initial frame.
             image_file = QtCore.QSettings().value(u'advanced/default image',
@@ -180,24 +200,6 @@
                 self.primary = True
         log.debug(u'Finished MainDisplay setup')
 
-    def createMediaObject(self):
-        self.firstTime = False
-        log.debug(u'Creating Phonon objects - Start for %s', self.isLive)
-        self.mediaObject = Phonon.MediaObject(self)
-        self.audio = Phonon.AudioOutput(Phonon.VideoCategory, self.mediaObject)
-        Phonon.createPath(self.mediaObject, self.videoWidget)
-        Phonon.createPath(self.mediaObject, self.audio)
-        QtCore.QObject.connect(self.mediaObject,
-            QtCore.SIGNAL(u'stateChanged(Phonon::State, Phonon::State)'),
-            self.videoState)
-        QtCore.QObject.connect(self.mediaObject,
-            QtCore.SIGNAL(u'finished()'),
-            self.videoFinished)
-        QtCore.QObject.connect(self.mediaObject,
-            QtCore.SIGNAL(u'tick(qint64)'),
-            self.videoTick)
-        log.debug(u'Creating Phonon objects - Finished for %s', self.isLive)
-
     def text(self, slide):
         """
         Add the slide text from slideController
@@ -221,8 +223,8 @@
             The text to be displayed.
         """
         log.debug(u'alert to display')
-        if self.height() != self.screen[u'size'].height() or not \
-            self.isVisible() or self.videoWidget.isVisible():
+        if self.height() != self.screen[u'size'].height() or \
+            not self.isVisible():
             shrink = True
             js = u'show_alert("%s", "%s")' % (
                 text.replace(u'\\', u'\\\\').replace(u'\"', u'\\\"'),
@@ -233,22 +235,18 @@
                 text.replace(u'\\', u'\\\\').replace(u'\"', u'\\\"'))
         height = self.frame.evaluateJavaScript(js)
         if shrink:
-            if self.phononActive:
-                shrinkItem = self.webView
-            else:
-                shrinkItem = self
             if text:
                 alert_height = int(height.toString())
-                shrinkItem.resize(self.width(), alert_height)
-                shrinkItem.setVisible(True)
+                self.resize(self.width(), alert_height)
+                self.setVisible(True)
                 if location == AlertLocation.Middle:
-                    shrinkItem.move(self.screen[u'size'].left(),
+                    self.move(self.screen[u'size'].left(),
                     (self.screen[u'size'].height() - alert_height) / 2)
                 elif location == AlertLocation.Bottom:
-                    shrinkItem.move(self.screen[u'size'].left(),
+                    self.move(self.screen[u'size'].left(),
                         self.screen[u'size'].height() - alert_height)
             else:
-                shrinkItem.setVisible(False)
+                self.setVisible(False)
                 self.setGeometry(self.screen[u'size'])
 
     def directImage(self, name, path, background):
@@ -276,7 +274,7 @@
         """
         log.debug(u'image to display')
         image = self.imageManager.get_image_bytes(name)
-        self.resetVideo()
+        self.controller.mediaController.video_reset(self.controller)
         self.displayImage(image)
 
     def displayImage(self, image):
@@ -303,128 +301,6 @@
         # clear the cache
         self.override = {}
 
-    def resetVideo(self):
-        """
-        Used after Video plugin has changed the background
-        """
-        log.debug(u'resetVideo')
-        if self.phononActive:
-            self.mediaObject.stop()
-            self.mediaObject.clearQueue()
-            self.webView.setVisible(True)
-            self.videoWidget.setVisible(False)
-            self.phononActive = False
-        else:
-            self.frame.evaluateJavaScript(u'show_video("close");')
-        self.override = {}
-
-    def videoPlay(self):
-        """
-        Responds to the request to play a loaded video
-        """
-        log.debug(u'videoPlay')
-        if self.phononActive:
-            self.mediaObject.play()
-        else:
-            self.frame.evaluateJavaScript(u'show_video("play");')
-        # show screen
-        if self.isLive:
-            self.setVisible(True)
-
-    def videoPause(self):
-        """
-        Responds to the request to pause a loaded video
-        """
-        log.debug(u'videoPause')
-        if self.phononActive:
-            self.mediaObject.pause()
-        else:
-            self.frame.evaluateJavaScript(u'show_video("pause");')
-
-    def videoStop(self):
-        """
-        Responds to the request to stop a loaded video
-        """
-        log.debug(u'videoStop')
-        if self.phononActive:
-            self.mediaObject.stop()
-        else:
-            self.frame.evaluateJavaScript(u'show_video("stop");')
-
-    def videoVolume(self, volume):
-        """
-        Changes the volume of a running video
-        """
-        log.debug(u'videoVolume %d' % volume)
-        vol = float(volume) / float(10)
-        if self.phononActive:
-            self.audio.setVolume(vol)
-        else:
-            self.frame.evaluateJavaScript(u'show_video(null, null, %s);' %
-                str(vol))
-
-    def video(self, videoPath, volume, isBackground=False):
-        """
-        Loads and starts a video to run with the option of sound
-        """
-        # We request a background video but have no service Item
-        if isBackground and not hasattr(self, u'serviceItem'):
-            return False
-        if not self.mediaObject:
-            self.createMediaObject()
-        log.debug(u'video')
-        self.webLoaded = True
-        self.setGeometry(self.screen[u'size'])
-        # We are running a background theme
-        self.override[u'theme'] = u''
-        self.override[u'video'] = True
-        vol = float(volume) / float(10)
-        if isBackground or not self.usePhonon:
-            js = u'show_video("init", "%s", %s, true); show_video("play");' % \
-                (videoPath.replace(u'\\', u'\\\\'), str(vol))
-            self.frame.evaluateJavaScript(js)
-        else:
-            self.phononActive = True
-            self.mediaObject.stop()
-            self.mediaObject.clearQueue()
-            self.mediaObject.setCurrentSource(Phonon.MediaSource(videoPath))
-            # Need the timer to trigger set the trigger to 200ms
-            # Value taken from web documentation.
-            if self.serviceItem.end_time != 0:
-                self.mediaObject.setTickInterval(200)
-            self.mediaObject.play()
-            self.webView.setVisible(False)
-            self.videoWidget.setVisible(True)
-            self.audio.setVolume(vol)
-        return True
-
-    def videoState(self, newState, oldState):
-        """
-        Start the video at a predetermined point.
-        """
-        if newState == Phonon.PlayingState \
-            and oldState != Phonon.PausedState \
-            and self.serviceItem.start_time > 0:
-            # set start time in milliseconds
-            self.mediaObject.seek(self.serviceItem.start_time * 1000)
-
-    def videoFinished(self):
-        """
-        Blank the Video when it has finished so the final frame is not left
-        hanging
-        """
-        self.videoStop()
-        self.hideDisplay(HideMode.Blank)
-        self.phononActive = False
-        self.videoHide = True
-
-    def videoTick(self, tick):
-        """
-        Triggered on video tick every 200 milli seconds
-        """
-        if tick > self.serviceItem.end_time * 1000:
-            self.videoFinished()
-
     def isWebLoaded(self):
         """
         Called by webView event to show display is fully loaded
@@ -511,16 +387,12 @@
         if serviceItem.foot_text:
             self.footer(serviceItem.foot_text)
         # if was hidden keep it hidden
-        if self.hideMode and self.isLive:
+        if self.hideMode and self.isLive and not serviceItem.is_media():
             if QtCore.QSettings().value(u'general/auto unblank',
                 QtCore.QVariant(False)).toBool():
                 Receiver.send_message(u'slidecontroller_live_unblank')
             else:
                 self.hideDisplay(self.hideMode)
-        # display hidden for video end we have a new item so must be shown
-        if self.videoHide and self.isLive:
-            self.videoHide = False
-            self.showDisplay()
         self.__hideMouse()
 
     def footer(self, text):
@@ -538,8 +410,6 @@
         Store the images so they can be replaced when required
         """
         log.debug(u'hideDisplay mode = %d', mode)
-        if self.phononActive:
-            self.videoPause()
         if mode == HideMode.Screen:
             self.frame.evaluateJavaScript(u'show_blank("desktop");')
             self.setVisible(False)
@@ -550,7 +420,6 @@
         if mode != HideMode.Screen:
             if self.isHidden():
                 self.setVisible(True)
-            if self.phononActive:
                 self.webView.setVisible(True)
         self.hideMode = mode
 
@@ -564,9 +433,6 @@
         self.frame.evaluateJavaScript('show_blank("show");')
         if self.isHidden():
             self.setVisible(True)
-        if self.phononActive:
-            self.webView.setVisible(False)
-            self.videoPlay()
         self.hideMode = None
         # Trigger actions when display is active again
         if self.isLive:

=== modified file 'openlp/core/ui/mainwindow.py'
--- openlp/core/ui/mainwindow.py	2011-10-22 11:09:01 +0000
+++ openlp/core/ui/mainwindow.py	2011-11-30 20:32:27 +0000
@@ -42,6 +42,7 @@
 from openlp.core.ui import AboutForm, SettingsForm, ServiceManager, \
     ThemeManager, SlideController, PluginForm, MediaDockManager, \
     ShortcutListForm, FormattingTagForm
+from openlp.core.ui.media import MediaController
 from openlp.core.utils import AppLocation, add_actions, LanguageManager, \
     get_application_version, delete_file
 from openlp.core.utils.actions import ActionList, CategoryOrder
@@ -557,6 +558,7 @@
         self.pluginManager = PluginManager(pluginpath)
         self.pluginHelpers = {}
         self.imageManager = ImageManager()
+        self.mediaController = MediaController(self)
         # Set up the interface
         self.setupUi(self)
         # Load settings after setupUi so default UI sizes are overwritten
@@ -644,6 +646,7 @@
         self.pluginHelpers[u'toolbox'] = self.mediaDockManager
         self.pluginHelpers[u'pluginmanager'] = self.pluginManager
         self.pluginHelpers[u'formparent'] = self
+        self.pluginHelpers[u'mediacontroller'] = self.mediaController
         self.pluginManager.find_plugins(pluginpath, self.pluginHelpers)
         # hook methods have to happen after find_plugins. Find plugins needs
         # the controllers hence the hooks have moved from setupUI() to here

=== added directory 'openlp/core/ui/media'
=== added file 'openlp/core/ui/media/__init__.py'
--- openlp/core/ui/media/__init__.py	1970-01-01 00:00:00 +0000
+++ openlp/core/ui/media/__init__.py	2011-11-30 20:32:27 +0000
@@ -0,0 +1,65 @@
+# -*- coding: utf-8 -*-
+# vim: autoindent shiftwidth=4 expandtab textwidth=80 tabstop=4 softtabstop=4
+
+###############################################################################
+# OpenLP - Open Source Lyrics Projection                                      #
+# --------------------------------------------------------------------------- #
+# Copyright (c) 2008-2011 Raoul Snyman                                        #
+# Portions copyright (c) 2008-2011 Tim Bentley, Gerald Britton, Jonathan      #
+# Corwin, Michael Gorven, Scott Guerrieri, Matthias Hub, Meinert Jordan,      #
+# Armin Köhler, Joshua Miller, Stevan Pettit, Andreas Preikschat, Mattias     #
+# Põldaru, Christian Richter, Philip Ridout, Simon Scudder, Jeffrey Smith,    #
+# Maikel Stuivenberg, Martin Thompson, Jon Tibble, Frode Woldsund             #
+# --------------------------------------------------------------------------- #
+# This program is free software; you can redistribute it and/or modify it     #
+# under the terms of the GNU General Public License as published by the Free  #
+# Software Foundation; version 2 of the License.                              #
+#                                                                             #
+# This program is distributed in the hope that it will be useful, but WITHOUT #
+# ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or       #
+# FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for    #
+# more details.                                                               #
+#                                                                             #
+# You should have received a copy of the GNU General Public License along     #
+# with this program; if not, write to the Free Software Foundation, Inc., 59  #
+# Temple Place, Suite 330, Boston, MA 02111-1307 USA                          #
+###############################################################################
+
+class MediaState(object):
+    """
+    An enumeration for possible States of the Media Player (copied partially 
+    from Phonon::State)
+    """
+    Loading = 0
+    Stopped = 1
+    Playing = 2
+    Paused = 4
+    Off = 6
+
+
+class MediaType(object):
+    """
+    An enumeration of possibible Media Types
+    """
+    Unused = 0
+    Audio = 1
+    Video = 2
+    CD = 3
+    DVD = 4
+    Folder = 5
+
+
+class MediaInfo(object):
+    """
+    This class hold the media related infos
+    """
+    file_info = None
+    volume = 100
+    is_flash = False
+    is_background = False
+    length = 0
+    start_time = 0
+    end_time = 0
+    media_type = MediaType()
+
+from mediacontroller import MediaController

=== added file 'openlp/core/ui/media/mediacontroller.py'
--- openlp/core/ui/media/mediacontroller.py	1970-01-01 00:00:00 +0000
+++ openlp/core/ui/media/mediacontroller.py	2011-11-30 20:32:27 +0000
@@ -0,0 +1,580 @@
+# -*- coding: utf-8 -*-
+# vim: autoindent shiftwidth=4 expandtab textwidth=80 tabstop=4 softtabstop=4
+
+###############################################################################
+# OpenLP - Open Source Lyrics Projection                                      #
+# --------------------------------------------------------------------------- #
+# Copyright (c) 2008-2011 Raoul Snyman                                        #
+# Portions copyright (c) 2008-2011 Tim Bentley, Gerald Britton, Jonathan      #
+# Corwin, Michael Gorven, Scott Guerrieri, Matthias Hub, Meinert Jordan,      #
+# Armin Köhler, Joshua Miller, Stevan Pettit, Andreas Preikschat, Mattias     #
+# Põldaru, Christian Richter, Philip Ridout, Simon Scudder, Jeffrey Smith,    #
+# Maikel Stuivenberg, Martin Thompson, Jon Tibble, Frode Woldsund             #
+# --------------------------------------------------------------------------- #
+# This program is free software; you can redistribute it and/or modify it     #
+# under the terms of the GNU General Public License as published by the Free  #
+# Software Foundation; version 2 of the License.                              #
+#                                                                             #
+# This program is distributed in the hope that it will be useful, but WITHOUT #
+# ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or       #
+# FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for    #
+# more details.                                                               #
+#                                                                             #
+# You should have received a copy of the GNU General Public License along     #
+# with this program; if not, write to the Free Software Foundation, Inc., 59  #
+# Temple Place, Suite 330, Boston, MA 02111-1307 USA                          #
+###############################################################################
+
+import logging
+
+import sys, os,time
+from PyQt4 import QtCore, QtGui, QtWebKit
+
+from openlp.core.lib import OpenLPToolbar, Receiver, translate
+from openlp.core.lib.mediaplayer import MediaPlayer
+from openlp.core.lib.ui import UiStrings, critical_error_message_box
+from openlp.core.ui.media import MediaState, MediaInfo, MediaType
+from openlp.core.utils import AppLocation
+
+log = logging.getLogger(__name__)
+
+class MediaController(object):
+    """
+    The implementation of the Media Controller. The Media Controller adds an own
+    class for every Player. Currently these are QtWebkit, Phonon and planed Vlc.
+    """
+
+    def __init__(self, parent):
+        self.parent = parent
+        self.mediaPlayers = {}
+        self.controller = []
+        self.overridenPlayer = ''
+        self.curDisplayMediaPlayer = {}
+        # Timer for video state
+        self.timer = QtCore.QTimer()
+        self.timer.setInterval(200)
+        self.withLivePreview = False
+        self.check_available_media_players()
+        # Signals
+        QtCore.QObject.connect(self.timer,
+            QtCore.SIGNAL("timeout()"), self.video_state)
+        QtCore.QObject.connect(Receiver.get_receiver(),
+            QtCore.SIGNAL(u'media_playback_play'), self.video_play)
+        QtCore.QObject.connect(Receiver.get_receiver(),
+            QtCore.SIGNAL(u'media_playback_pause'), self.video_pause)
+        QtCore.QObject.connect(Receiver.get_receiver(),
+            QtCore.SIGNAL(u'media_playback_stop'), self.video_stop)
+        QtCore.QObject.connect(Receiver.get_receiver(),
+            QtCore.SIGNAL(u'seek_slider'), self.video_seek)
+        QtCore.QObject.connect(Receiver.get_receiver(),
+            QtCore.SIGNAL(u'volume_slider'), self.video_volume)
+        QtCore.QObject.connect(Receiver.get_receiver(),
+            QtCore.SIGNAL(u'media_hide'), self.video_hide)
+        QtCore.QObject.connect(Receiver.get_receiver(),
+            QtCore.SIGNAL(u'media_blank'), self.video_blank)
+        QtCore.QObject.connect(Receiver.get_receiver(),
+            QtCore.SIGNAL(u'media_unblank'), self.video_unblank)
+        QtCore.QObject.connect(Receiver.get_receiver(),
+            QtCore.SIGNAL(u'media_override_player'), self.override_player)
+        # Signals for background video
+        QtCore.QObject.connect(Receiver.get_receiver(),
+            QtCore.SIGNAL(u'songs_hide'), self.video_hide)
+        QtCore.QObject.connect(Receiver.get_receiver(),
+            QtCore.SIGNAL(u'songs_unblank'), self.video_unblank)
+        QtCore.QObject.connect(Receiver.get_receiver(),
+            QtCore.SIGNAL(u'mediaitem_media_rebuild'), self.set_active_players)
+
+    def set_active_players(self):
+        playerSettings = str(QtCore.QSettings().value(u'media/players',
+            QtCore.QVariant(u'webkit')).toString())
+        if len(playerSettings) == 0:
+            playerSettings = u'webkit'
+        savedPlayers = playerSettings.split(u',')
+        for player in self.mediaPlayers.keys():
+            if player in savedPlayers:
+                self.mediaPlayers[player].isActive = True
+            else:
+                self.mediaPlayers[player].isActive = False
+
+    def register_controllers(self, controller):
+        """
+        Register each media Player controller (Webkit, Phonon, etc) and store
+        for later use
+        """
+        if controller.check_available():
+            self.mediaPlayers[controller.name] = controller
+
+    def check_available_media_players(self):
+        """
+        Check to see if we have any media Player's available. If Not do not
+        install the plugin.
+        """
+        log.debug(u'check_available_media_players')
+        controller_dir = os.path.join(
+            AppLocation.get_directory(AppLocation.AppDir),
+            u'core', u'ui', u'media')
+        for filename in os.listdir(controller_dir):
+            if filename.endswith(u'player.py') and \
+                not filename == 'media_player.py':
+                path = os.path.join(controller_dir, filename)
+                if os.path.isfile(path):
+                    modulename = u'openlp.core.ui.media.' + \
+                        os.path.splitext(filename)[0]
+                    log.debug(u'Importing controller %s', modulename)
+                    try:
+                        __import__(modulename, globals(), locals(), [])
+                    except ImportError:
+                        log.warn(u'Failed to import %s on path %s',
+                            modulename, path)
+        controller_classes = MediaPlayer.__subclasses__()
+        for controller_class in controller_classes:
+            controller = controller_class(self)
+            self.register_controllers(controller)
+        if self.mediaPlayers:
+            playerSettings = str(QtCore.QSettings().value(u'media/players',
+                QtCore.QVariant(u'webkit')).toString())
+            savedPlayers = playerSettings.split(u',')
+            invalidMediaPlayers = [mediaPlayer for mediaPlayer in savedPlayers \
+                if not mediaPlayer in self.mediaPlayers]
+            if len(invalidMediaPlayers)>0:
+                [savedPlayers.remove(invalidPlayer) for invalidPlayer in invalidMediaPlayers]
+                newPlayerSetting = u','.join(savedPlayers)
+                QtCore.QSettings().setValue(u'media/players',
+                    QtCore.QVariant(newPlayerSetting))
+            self.set_active_players()
+            return True
+        else:
+            return False
+
+    def video_state(self):
+        """
+        Check if there is a running media Player and do updating stuff (e.g.
+        update the UI)
+        """
+        if len(self.curDisplayMediaPlayer.keys()) == 0:
+            self.timer.stop()
+        else:
+            for display in self.curDisplayMediaPlayer.keys():
+                self.curDisplayMediaPlayer[display].resize(display)
+                self.curDisplayMediaPlayer[display].update_ui(display)
+                if self.curDisplayMediaPlayer[display] \
+                    .state == MediaState.Playing:
+                    return
+        self.timer.stop()
+
+    def get_media_display_css(self):
+        """
+        Add css style sheets to htmlbuilder
+        """
+        css = u''
+        for player in self.mediaPlayers.values():
+            if player.isActive:
+                css += player.get_media_display_css()
+        return css
+
+    def get_media_display_javascript(self):
+        """
+        Add javascript functions to htmlbuilder
+        """
+        js = u''
+        for player in self.mediaPlayers.values():
+            if player.isActive:
+                js += player.get_media_display_javascript()
+        return js
+
+    def get_media_display_html(self):
+        """
+        Add html code to htmlbuilder
+        """
+        html = u''
+        for player in self.mediaPlayers.values():
+            if player.isActive:
+                html += player.get_media_display_html()
+        return html
+
+    def add_controller_items(self, controller, control_panel):
+        self.controller.append(controller)
+        self.setup_generic_controls(controller, control_panel)
+        self.setup_special_controls(controller, control_panel)
+
+    def setup_generic_controls(self, controller, control_panel):
+        """
+        Add generic media control items (valid for all types of medias)
+        """
+        controller.media_info = MediaInfo()
+        # Build a Media ToolBar
+        controller.mediabar = OpenLPToolbar(controller)
+        controller.mediabar.addToolbarButton(
+            u'media_playback_play', u':/slides/media_playback_start.png',
+            translate('OpenLP.SlideController', 'Start playing media'),
+            controller.sendToPlugins)
+        controller.mediabar.addToolbarButton(
+            u'media_playback_pause', u':/slides/media_playback_pause.png',
+            translate('OpenLP.SlideController', 'Pause playing media'),
+            controller.sendToPlugins)
+        controller.mediabar.addToolbarButton(
+            u'media_playback_stop', u':/slides/media_playback_stop.png',
+            translate('OpenLP.SlideController', 'Stop playing media'),
+            controller.sendToPlugins)
+        # Build the seekSlider.
+        controller.seekSlider = QtGui.QSlider(QtCore.Qt.Horizontal)
+        controller.seekSlider.setMaximum(1000)
+        controller.seekSlider.setToolTip(translate(
+            'OpenLP.SlideController', 'Video position.'))
+        controller.seekSlider.setGeometry(QtCore.QRect(90, 260, 221, 24))
+        controller.seekSlider.setObjectName(u'seek_slider')
+        controller.mediabar.addToolbarWidget(u'Seek Slider', 
+            controller.seekSlider)
+        # Build the volumeSlider.
+        controller.volumeSlider = QtGui.QSlider(QtCore.Qt.Horizontal)
+        controller.volumeSlider.setTickInterval(10)
+        controller.volumeSlider.setTickPosition(QtGui.QSlider.TicksAbove)
+        controller.volumeSlider.setMinimum(0)
+        controller.volumeSlider.setMaximum(100)
+        controller.volumeSlider.setToolTip(translate(
+            'OpenLP.SlideController', 'Audio Volume.'))
+        controller.volumeSlider.setValue(controller.media_info.volume)
+        controller.volumeSlider.setGeometry(QtCore.QRect(90, 160, 221, 24))
+        controller.volumeSlider.setObjectName(u'volume_slider')
+        controller.mediabar.addToolbarWidget(u'Audio Volume', 
+            controller.volumeSlider)
+        control_panel.addWidget(controller.mediabar)
+        controller.mediabar.setVisible(False)
+        # Signals
+        QtCore.QObject.connect(controller.seekSlider,
+            QtCore.SIGNAL(u'sliderMoved(int)'), controller.sendToPlugins)
+        QtCore.QObject.connect(controller.volumeSlider,
+            QtCore.SIGNAL(u'sliderMoved(int)'), controller.sendToPlugins)
+
+    def setup_special_controls(self, controller, control_panel):
+        """
+        Special media Toolbars will be created here (e.g. for DVD Playback)
+        """
+        controller.media_info = MediaInfo()
+        # TODO: add Toolbar for DVD, ...
+
+    def setup_display(self, display):
+        """
+        After a new display is configured, all media related widget will be 
+        created too
+        """
+        # clean up possible running old media files
+        self.finalise()
+        # update player status
+        self.set_active_players()
+        display.hasAudio = True
+        if not self.withLivePreview and \
+            display == self.parent.liveController.previewDisplay:
+            return
+        if display == self.parent.previewController.previewDisplay or \
+            display == self.parent.liveController.previewDisplay:
+            display.hasAudio = False
+        for player in self.mediaPlayers.values():
+            if player.isActive:
+                player.setup(display)
+
+    def set_controls_visible(self, controller, value):
+        # Generic controls
+        controller.mediabar.setVisible(value)
+        # Special controls: Here media type specific Controls will be enabled 
+        # (e.g. for DVD control, ...)
+        # TODO
+
+    def resize(self, controller, display, player):
+        """
+        After Mainwindow changes or Splitter moved all related media widgets 
+        have to be resized
+        """
+        player.resize(display)
+
+    def video(self, controller, file, muted, isBackground):
+        """
+        Loads and starts a video to run with the option of sound
+        """
+        log.debug(u'video')
+        isValid = False
+        # stop running videos
+        self.video_reset(controller)
+        controller.media_info = MediaInfo()
+        if muted:
+            controller.media_info.volume = 0
+        else:
+            controller.media_info.volume = controller.volumeSlider.value()
+        controller.media_info.file_info = QtCore.QFileInfo(file)
+        controller.media_info.is_background = isBackground
+        display = None
+        if controller.isLive:
+            if self.withLivePreview and controller.previewDisplay:
+                display = controller.previewDisplay
+                isValid = self.check_file_type(controller, display)
+            display = controller.display
+            isValid = self.check_file_type(controller, display)
+            display.override[u'theme'] = u''
+            display.override[u'video'] = True
+            controller.media_info.start_time = display.serviceItem.start_time
+            controller.media_info.end_time = display.serviceItem.end_time
+        elif controller.previewDisplay:
+            display = controller.previewDisplay
+            isValid = self.check_file_type(controller, display)
+        if not isValid:
+            # Media could not be loaded correctly
+            critical_error_message_box(
+                translate('MediaPlugin.MediaItem', 'Unsupported File'),
+                unicode(translate('MediaPlugin.MediaItem',
+                'Unsupported File')))
+            return False
+        # dont care about actual theme, set a black background
+        if controller.isLive and ( \
+            controller.media_info.is_background == False):
+            display.frame.evaluateJavaScript(u'show_video( \
+            "setBackBoard", null, null, null,"visible");')
+        # now start playing
+        if self.video_play([controller], False):
+            self.video_pause([controller])
+            self.video_seek([controller, [0]])
+            if controller.isLive and \
+                (QtCore.QSettings().value(u'general/auto unblank',
+                QtCore.QVariant(False)).toBool() or \
+                controller.media_info.is_background == True) or \
+                controller.isLive == False:
+                self.video_play([controller])
+            self.set_controls_visible(controller, True)
+            log.debug(u'use %s controller' % self.curDisplayMediaPlayer[display])
+            return True
+        else:
+            critical_error_message_box(
+                translate('MediaPlugin.MediaItem', 'Unsupported File'),
+                unicode(translate('MediaPlugin.MediaItem',
+                'Unsupported File')))
+        return False
+
+    def check_file_type(self, controller, display):
+        """
+        Used to choose the right media Player type from the prioritized Player list
+        """
+        playerSettings = str(QtCore.QSettings().value(u'media/players',
+            QtCore.QVariant(u'webkit')).toString())
+        usedPlayers = playerSettings.split(u',')
+        if QtCore.QSettings().value(u'media/override player',
+            QtCore.QVariant(QtCore.Qt.Unchecked)) == QtCore.Qt.Checked:
+            if self.overridenPlayer != '':
+                usedPlayers = [self.overridenPlayer]
+        if controller.media_info.file_info.isFile():
+            suffix = u'*.%s' % controller.media_info.file_info.suffix().toLower()
+            for title in usedPlayers:
+                player = self.mediaPlayers[title]
+                if suffix in player.video_extensions_list:
+                    if not controller.media_info.is_background or \
+                        controller.media_info.is_background and player.canBackground:
+                            self.resize(controller, display, player)
+                            if player.load(display):
+                                self.curDisplayMediaPlayer[display] = player
+                                controller.media_info.media_type = MediaType.Video
+                                return True
+                if suffix in player.audio_extensions_list:
+                    if player.load(display):
+                        self.curDisplayMediaPlayer[display] = player
+                        controller.media_info.media_type = MediaType.Audio
+                        return True
+        else:
+            for title in usedPlayers:
+                player = self.mediaPlayers[title]
+                if player.canFolder:
+                    self.resize(controller, display, player)
+                    if player.load(display):
+                        self.curDisplayMediaPlayer[display] = player
+                        controller.media_info.media_type = MediaType.Video
+                        return True
+        # no valid player found
+        return False
+
+    def video_play(self, msg, status=True):
+        """
+        Responds to the request to play a loaded video
+        
+         ``msg``
+            First element is the controller which should be used
+        """
+        log.debug(u'video_play')
+        controller = msg[0]
+        for display in self.curDisplayMediaPlayer.keys():
+            if display.controller == controller:
+                if not self.curDisplayMediaPlayer[display].play(display):
+                    return False
+                if status:
+                    display.frame.evaluateJavaScript(u'show_blank("desktop");')
+                    self.curDisplayMediaPlayer[display].set_visible(display, True)
+                    if controller.isLive:
+                        if controller.hideMenu.defaultAction().isChecked():
+                            controller.hideMenu.defaultAction().trigger()
+        # Start Timer for ui updates
+        if not self.timer.isActive():
+            self.timer.start()
+        return True
+
+    def video_pause(self, msg):
+        """
+        Responds to the request to pause a loaded video
+
+        ``msg``
+            First element is the controller which should be used
+        """
+        log.debug(u'video_pause')
+        controller = msg[0]
+        for display in self.curDisplayMediaPlayer.keys():
+            if display.controller == controller:
+                self.curDisplayMediaPlayer[display].pause(display)
+
+    def video_stop(self, msg):
+        """
+        Responds to the request to stop a loaded video
+
+        ``msg``
+            First element is the controller which should be used
+        """
+        log.debug(u'video_stop')
+        controller = msg[0]
+        for display in self.curDisplayMediaPlayer.keys():
+            if display.controller == controller:
+                display.frame.evaluateJavaScript(u'show_blank("black");')
+                self.curDisplayMediaPlayer[display].stop(display)
+                self.curDisplayMediaPlayer[display].set_visible(display, False)
+
+    def video_volume(self, msg):
+        """
+        Changes the volume of a running video
+
+        ``msg``
+            First element is the controller which should be used
+        """
+        controller = msg[0]
+        vol = msg[1][0]
+        log.debug(u'video_volume %d' % vol)
+        for display in self.curDisplayMediaPlayer.keys():
+            if display.controller == controller:
+                self.curDisplayMediaPlayer[display].volume(display, vol)
+
+    def video_seek(self, msg):
+        """
+        Responds to the request to change the seek Slider of a loaded video
+
+        ``msg``
+            First element is the controller which should be used
+            Second element is a list with the seek Value as first element
+        """
+        log.debug(u'video_seek')
+        controller = msg[0]
+        seekVal = msg[1][0]
+        for display in self.curDisplayMediaPlayer.keys():
+            if display.controller == controller:
+                self.curDisplayMediaPlayer[display].seek(display, seekVal)
+
+    def video_reset(self, controller):
+        """
+        Responds to the request to reset a loaded video
+        """
+        log.debug(u'video_reset')
+        for display in self.curDisplayMediaPlayer.keys():
+            if display.controller == controller:
+                display.override = {}
+                self.curDisplayMediaPlayer[display].reset(display)
+                self.curDisplayMediaPlayer[display].set_visible(display, False)
+                display.frame.evaluateJavaScript(u'show_video( \
+                "setBackBoard", null, null, null,"hidden");')
+                del self.curDisplayMediaPlayer[display]
+        self.set_controls_visible(controller, False)
+
+    def video_hide(self, msg):
+        """
+        Hide the related video Widget
+
+        ``msg``
+            First element is the boolean for Live indication
+        """
+        isLive = msg[1]
+        if isLive:
+            controller = self.parent.liveController
+            for display in self.curDisplayMediaPlayer.keys():
+                if display.controller == controller:
+                    if self.curDisplayMediaPlayer[display] \
+                        .state == MediaState.Playing:
+                        self.curDisplayMediaPlayer[display].pause(display)
+                        self.curDisplayMediaPlayer[display] \
+                            .set_visible(display, False)
+
+    def video_blank(self, msg):
+        """
+        Blank the related video Widget
+
+        ``msg``
+            First element is the boolean for Live indication
+            Second element is the hide mode
+        """
+        isLive = msg[1]
+        hide_mode = msg[2]
+        if isLive:
+            Receiver.send_message(u'live_display_hide', hide_mode)
+            controller = self.parent.liveController
+            for display in self.curDisplayMediaPlayer.keys():
+                if display.controller == controller:
+                    if self.curDisplayMediaPlayer[display] \
+                        .state == MediaState.Playing:
+                        self.curDisplayMediaPlayer[display].pause(display)
+                        self.curDisplayMediaPlayer[display] \
+                            .set_visible(display, False)
+
+    def video_unblank(self, msg):
+        """
+        Unblank the related video Widget
+
+        ``msg``
+            First element is not relevant in this context
+            Second element is the boolean for Live indication
+        """
+        Receiver.send_message(u'live_display_show')
+        isLive = msg[1]
+        if isLive:
+            controller = self.parent.liveController
+            for display in self.curDisplayMediaPlayer.keys():
+                if display.controller == controller:
+                    if self.curDisplayMediaPlayer[display] \
+                        .state == MediaState.Paused:
+                        if self.curDisplayMediaPlayer[display].play(display):
+                            self.curDisplayMediaPlayer[display] \
+                                .set_visible(display, True)
+                                    # Start Timer for ui updates
+                            if not self.timer.isActive():
+                                self.timer.start()
+
+
+    def get_audio_extensions_list(self):
+        audio_list = []
+        for player in self.mediaPlayers.values():
+            if player.isActive:
+                for item in player.audio_extensions_list:
+                    if not item in audio_list:
+                        audio_list.append(item)
+        return audio_list
+
+    def get_video_extensions_list(self):
+        video_list = []
+        for player in self.mediaPlayers.values():
+            if player.isActive:
+                for item in player.video_extensions_list:
+                    if not item in video_list:
+                        video_list.append(item)
+        return video_list
+
+    def override_player(self, override_player):
+        playerSettings = str(QtCore.QSettings().value(u'media/players',
+            QtCore.QVariant(u'webkit')).toString())
+        usedPlayers = playerSettings.split(u',')
+        if override_player in usedPlayers:
+            self.overridenPlayer = override_player
+        else:
+            self.overridenPlayer = ''
+
+    def finalise(self):
+        self.timer.stop()
+        for controller in self.controller:
+            self.video_reset(controller)

=== added file 'openlp/core/ui/media/phononplayer.py'
--- openlp/core/ui/media/phononplayer.py	1970-01-01 00:00:00 +0000
+++ openlp/core/ui/media/phononplayer.py	2011-11-30 20:32:27 +0000
@@ -0,0 +1,201 @@
+# -*- coding: utf-8 -*-
+# vim: autoindent shiftwidth=4 expandtab textwidth=80 tabstop=4 softtabstop=4
+
+###############################################################################
+# OpenLP - Open Source Lyrics Projection                                      #
+# --------------------------------------------------------------------------- #
+# Copyright (c) 2008-2011 Raoul Snyman                                        #
+# Portions copyright (c) 2008-2011 Tim Bentley, Gerald Britton, Jonathan      #
+# Corwin, Michael Gorven, Scott Guerrieri, Matthias Hub, Meinert Jordan,      #
+# Armin Köhler, Joshua Miller, Stevan Pettit, Andreas Preikschat, Mattias     #
+# Põldaru, Christian Richter, Philip Ridout, Simon Scudder, Jeffrey Smith,    #
+# Maikel Stuivenberg, Martin Thompson, Jon Tibble, Frode Woldsund             #
+# --------------------------------------------------------------------------- #
+# This program is free software; you can redistribute it and/or modify it     #
+# under the terms of the GNU General Public License as published by the Free  #
+# Software Foundation; version 2 of the License.                              #
+#                                                                             #
+# This program is distributed in the hope that it will be useful, but WITHOUT #
+# ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or       #
+# FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for    #
+# more details.                                                               #
+#                                                                             #
+# You should have received a copy of the GNU General Public License along     #
+# with this program; if not, write to the Free Software Foundation, Inc., 59  #
+# Temple Place, Suite 330, Boston, MA 02111-1307 USA                          #
+###############################################################################
+
+import logging
+import mimetypes
+from datetime import datetime
+
+from PyQt4 import QtCore, QtGui
+from PyQt4.phonon import Phonon
+
+from openlp.core.lib import Receiver
+from openlp.core.lib.mediaplayer import MediaPlayer
+from openlp.core.ui.media import MediaState
+
+log = logging.getLogger(__name__)
+
+ADDITIONAL_EXT = {
+            u'audio/ac3': [u'.ac3'],
+            u'audio/flac': [u'.flac'],
+            u'audio/x-m4a': [u'.m4a'],
+            u'audio/midi': [u'.mid', u'.midi'],
+            u'audio/x-mp3': [u'.mp3'],
+            u'audio/mpeg': [u'.mp3', u'.mp2', u'.mpga', u'.mpega', u'.m4a'],
+            u'audio/qcelp': [u'.qcp'],
+            u'audio/x-wma': [u'.wma'],
+            u'audio/x-ms-wma': [u'.wma'],
+            u'video/x-flv': [u'.flv'],
+            u'video/x-matroska': [u'.mpv', u'.mkv'],
+            u'video/x-wmv': [u'.wmv'],
+            u'video/x-mpg': [u'.mpg'],
+            u'video/x-ms-wmv': [u'.wmv']}
+
+
+class PhononPlayer(MediaPlayer):
+    """
+    A specialised version of the MediaPlayer class, which provides a Phonon 
+    display.
+    """
+
+    def __init__(self, parent):
+        MediaPlayer.__init__(self, parent, u'phonon')
+        self.parent = parent
+        self.additional_extensions = ADDITIONAL_EXT
+        mimetypes.init()
+        for mimetype in Phonon.BackendCapabilities.availableMimeTypes():
+            mimetype = unicode(mimetype)
+            if mimetype.startswith(u'audio/'):
+                self._addToList(self.audio_extensions_list, mimetype)
+            elif mimetype.startswith(u'video/'):
+                self._addToList(self.video_extensions_list, mimetype)
+
+    def _addToList(self, list, mimetype):
+        # Add all extensions which mimetypes provides us for supported types.
+        extensions = mimetypes.guess_all_extensions(unicode(mimetype))
+        for extension in extensions:
+            ext = u'*%s' % extension
+            if ext not in list:
+                list.append(ext)
+        log.info(u'MediaPlugin: %s extensions: %s' % (mimetype,
+            u' '.join(extensions)))
+        # Add extensions for this mimetype from self.additional_extensions.
+        # This hack clears mimetypes' and operating system's shortcomings
+        # by providing possibly missing extensions.
+        if mimetype in self.additional_extensions.keys():
+            for extension in self.additional_extensions[mimetype]:
+                ext = u'*%s' % extension
+                if ext not in list:
+                    list.append(ext)
+            log.info(u'MediaPlugin: %s additional extensions: %s' % (mimetype,
+                u' '.join(self.additional_extensions[mimetype])))
+
+    def setup(self, display):
+        display.phononWidget = Phonon.VideoWidget(display)
+        display.phononWidget.resize(display.size())
+        display.mediaObject = Phonon.MediaObject(display)
+        Phonon.createPath(display.mediaObject, display.phononWidget)
+        if display.hasAudio:
+            display.audio = Phonon.AudioOutput( \
+                Phonon.VideoCategory, display.mediaObject)
+            Phonon.createPath(display.mediaObject, display.audio)
+        display.phononWidget.raise_()
+        display.phononWidget.hide()
+        self.hasOwnWidget = True
+
+    def check_available(self):
+        return True
+
+    def load(self, display):
+        log.debug(u'load vid in Phonon Controller')
+        controller = display.controller
+        volume = controller.media_info.volume
+        path = controller.media_info.file_info.absoluteFilePath()
+        display.mediaObject.setCurrentSource(Phonon.MediaSource(path))
+        if not self.media_state_wait(display, Phonon.StoppedState):
+            return False
+        self.volume(display, volume)
+        return True
+
+    def media_state_wait(self, display, mediaState):
+        """
+        Wait for the video to change its state
+        Wait no longer than 5 seconds.
+        """
+        start = datetime.now()
+        current_state = display.mediaObject.state()
+        while current_state != mediaState:
+            current_state = display.mediaObject.state()
+            if current_state == Phonon.ErrorState:
+                return False
+            Receiver.send_message(u'openlp_process_events')
+            if (datetime.now() - start).seconds > 5:
+                return False
+        return True
+
+    def resize(self, display):
+        display.phononWidget.resize(display.size())
+
+    def play(self, display):
+        controller = display.controller
+        start_time = 0
+        if display.mediaObject.state() != Phonon.PausedState and \
+            controller.media_info.start_time > 0:
+            start_time = controller.media_info.start_time
+        display.mediaObject.play()
+        if self.media_state_wait(display, Phonon.PlayingState):
+            if start_time > 0:
+                self.seek(display, controller.media_info.start_time*1000)
+            self.volume(display, controller.media_info.volume)
+            controller.media_info.length = \
+                int(display.mediaObject.totalTime()/1000)
+            controller.seekSlider.setMaximum(controller.media_info.length*1000)
+            self.state = MediaState.Playing
+            display.phononWidget.raise_()            
+            return True
+        else:
+            return False
+
+    def pause(self, display):
+        display.mediaObject.pause()
+        if self.media_state_wait(display, Phonon.PausedState):
+            self.state = MediaState.Paused
+
+    def stop(self, display):
+        display.mediaObject.stop()
+        self.set_visible(display, False)
+        self.state = MediaState.Stopped
+
+    def volume(self, display, vol):
+        # 1.0 is the highest value
+        if display.hasAudio:
+            vol = float(vol) / float(100)
+            display.audio.setVolume(vol)
+
+    def seek(self, display, seekVal):
+        display.mediaObject.seek(seekVal)
+
+    def reset(self, display):
+        display.mediaObject.stop()
+        display.mediaObject.clearQueue()
+        self.set_visible(display, False)
+        display.phononWidget.setVisible(False)
+        self.state = MediaState.Off
+
+    def set_visible(self, display, status):
+        if self.hasOwnWidget:
+            display.phononWidget.setVisible(status)
+
+    def update_ui(self, display):
+        controller = display.controller
+        if controller.media_info.end_time > 0:
+            if display.mediaObject.currentTime() > \
+                controller.media_info.end_time*1000:
+                self.stop(display)
+                self.set_visible(display, False)
+        if not controller.seekSlider.isSliderDown():
+            controller.seekSlider.setSliderPosition( \
+                display.mediaObject.currentTime())

=== added file 'openlp/core/ui/media/webkitplayer.py'
--- openlp/core/ui/media/webkitplayer.py	1970-01-01 00:00:00 +0000
+++ openlp/core/ui/media/webkitplayer.py	2011-11-30 20:32:27 +0000
@@ -0,0 +1,420 @@
+# -*- coding: utf-8 -*-
+# vim: autoindent shiftwidth=4 expandtab textwidth=80 tabstop=4 softtabstop=4
+
+###############################################################################
+# OpenLP - Open Source Lyrics Projection                                      #
+# --------------------------------------------------------------------------- #
+# Copyright (c) 2008-2011 Raoul Snyman                                        #
+# Portions copyright (c) 2008-2011 Tim Bentley, Gerald Britton, Jonathan      #
+# Corwin, Michael Gorven, Scott Guerrieri, Matthias Hub, Meinert Jordan,      #
+# Armin Köhler, Joshua Miller, Stevan Pettit, Andreas Preikschat, Mattias     #
+# Põldaru, Christian Richter, Philip Ridout, Simon Scudder, Jeffrey Smith,    #
+# Maikel Stuivenberg, Martin Thompson, Jon Tibble, Frode Woldsund             #
+# --------------------------------------------------------------------------- #
+# This program is free software; you can redistribute it and/or modify it     #
+# under the terms of the GNU General Public License as published by the Free  #
+# Software Foundation; version 2 of the License.                              #
+#                                                                             #
+# This program is distributed in the hope that it will be useful, but WITHOUT #
+# ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or       #
+# FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for    #
+# more details.                                                               #
+#                                                                             #
+# You should have received a copy of the GNU General Public License along     #
+# with this program; if not, write to the Free Software Foundation, Inc., 59  #
+# Temple Place, Suite 330, Boston, MA 02111-1307 USA                          #
+###############################################################################
+
+import logging
+
+from PyQt4 import QtCore, QtGui, QtWebKit
+
+from openlp.core.lib import OpenLPToolbar, translate
+from openlp.core.lib.mediaplayer import MediaPlayer
+from openlp.core.ui.media import MediaState
+
+log = logging.getLogger(__name__)
+
+VIDEO_CSS = u"""
+#videobackboard {
+    z-index:3;
+    background-color: black;
+}
+#video1 {
+    z-index:4;
+}
+#video2 {
+    z-index:4;
+}
+"""
+
+VIDEO_JS = u"""
+    var video_timer = null;
+    var current_video = '1';
+
+    function show_video(state, path, volume, loop, varVal){
+        // Note, the preferred method for looping would be to use the
+        // video tag loop attribute.
+        // But QtWebKit doesn't support this. Neither does it support the
+        // onended event, hence the setInterval()
+        // In addition, setting the currentTime attribute to zero to restart
+        // the video raises an INDEX_SIZE_ERROR: DOM Exception 1
+        // To complicate it further, sometimes vid.currentTime stops
+        // slightly short of vid.duration and vid.ended is intermittent!
+        //
+        // Note, currently the background may go black between loops. Not
+        // desirable. Need to investigate using two <video>'s, and hiding/
+        // preloading one, and toggle between the two when looping.
+
+        if(current_video=='1'){
+            var vid = document.getElementById('video1');
+            var vid2 = document.getElementById('video2');
+        } else {
+            var vid = document.getElementById('video2');
+            var vid2 = document.getElementById('video1');
+        }
+        if(volume != null){
+            vid.volume = volume;
+            vid2.volume = volume;
+        }
+        switch(state){
+            case 'init':
+                vid.src = 'file:///' + path;
+                vid2.src = 'file:///' + path;
+                if(loop == null) loop = false;
+                vid.looping = loop;
+                vid2.looping = loop;
+                vid.load();
+                break;
+            case 'load':
+                vid2.style.visibility = 'hidden';
+                vid2.load();
+                break;
+            case 'play':
+                vid.play();
+                if(vid.looping){
+                    video_timer = setInterval(
+                        function() {
+                            show_video('poll');
+                        }, 200);
+                }
+                break;
+            case 'pause':
+                if(video_timer!=null){
+                    clearInterval(video_timer);
+                    video_timer = null;
+                }
+                vid.pause();
+                break;
+            case 'stop':
+                show_video('pause');
+                vid.currentTime = 0;
+                break;
+            case 'poll':
+                if(vid.ended||vid.currentTime+0.2>vid.duration)
+                    show_video('swap');
+                break;
+            case 'swap':
+                show_video('pause');
+                if(current_video=='1')
+                    current_video = '2';
+                else
+                    current_video = '1';
+                show_video('load');
+                show_video('play');
+                show_video('setVisible',null,null,null,'visible');
+                break;
+            case 'close':
+                show_video('stop');
+                vid.src = '';
+                vid2.src = '';
+                break;
+             case 'length':
+                return vid.duration;
+            case 'currentTime':
+                return vid.currentTime;
+            case 'seek':
+                // doesnt work currently
+                vid.currentTime = varVal;
+                break;
+            case 'setVisible':
+                vid.style.visibility = varVal;
+                break;
+            case 'setBackBoard':
+                var back = document.getElementById('videobackboard');
+                back.style.visibility = varVal;
+                break;
+       }
+    }
+"""
+
+VIDEO_HTML = u"""
+<div id="videobackboard" class="size" style="visibility:hidden"></div>
+<video id="video1" class="size" style="visibility:hidden" autobuffer preload>
+</video>
+<video id="video2" class="size" style="visibility:hidden" autobuffer preload>
+</video>
+"""
+
+FLASH_CSS = u"""
+#flash {
+    z-index:5;
+}
+"""
+
+FLASH_JS = u"""
+    function getFlashMovieObject(movieName)
+    {
+        if (window.document[movieName])
+        {
+            return window.document[movieName];
+        }
+        if (document.embeds && document.embeds[movieName])
+            return document.embeds[movieName];
+    }
+
+    function show_flash(state, path, volume, varVal){
+        var text = document.getElementById('flash');
+        var flashMovie = getFlashMovieObject("OpenLPFlashMovie");
+        var src = "src = 'file:///" + path + "'";
+        var view_parm = " wmode='opaque'" +
+            " width='100%%'" +
+            " height='100%%'";
+        var swf_parm = " name='OpenLPFlashMovie'" +
+            " autostart='true' loop='false' play='true'" +
+            " hidden='false' swliveconnect='true' allowscriptaccess='always'" +
+            " volume='" + volume + "'";
+
+        switch(state){
+            case 'load':
+                text.innerHTML = "<embed " + src + view_parm + swf_parm + "/>";
+                flashMovie = getFlashMovieObject("OpenLPFlashMovie");
+                flashMovie.Play();
+                break;
+            case 'play':
+                flashMovie.Play();
+                break;
+            case 'pause':
+                flashMovie.StopPlay();
+                break;
+            case 'stop':
+                flashMovie.StopPlay();
+                tempHtml = text.innerHTML;
+                text.innerHTML = '';
+                text.innerHTML = tempHtml;
+                break;
+            case 'close':
+                flashMovie.StopPlay();
+                text.innerHTML = '';
+                break;
+            case 'length':
+                return flashMovie.TotalFrames();
+            case 'currentTime':
+                return flashMovie.CurrentFrame();
+            case 'seek':
+//                flashMovie.GotoFrame(varVal);
+                break;
+            case 'setVisible':
+                text.style.visibility = varVal;
+                break;
+        }
+    }
+"""
+
+FLASH_HTML = u"""
+<div id="flash" class="size" style="visibility:hidden"></div>
+"""
+
+VIDEO_EXT = [
+             u'*.3gp'
+            , u'*.3gpp'
+            , u'*.3g2'
+            , u'*.3gpp2'
+            , u'*.aac'
+            , u'*.flv'
+            , u'*.f4a'
+            , u'*.f4b'
+            , u'*.f4p'
+            , u'*.f4v'
+            , u'*.mov'
+            , u'*.m4a'
+            , u'*.m4b'
+            , u'*.m4p'
+            , u'*.m4v'
+            , u'*.mkv'
+            , u'*.mp4'
+            , u'*.ogv'
+            , u'*.webm'
+            , u'*.mpg', u'*.wmv',  u'*.mpeg', u'*.avi'
+            , u'*.swf'
+        ]
+
+AUDIO_EXT =  [
+              u'*.mp3'
+            , u'*.ogg'
+        ]
+
+
+class WebkitPlayer(MediaPlayer):
+    """
+    A specialised version of the MediaPlayer class, which provides a QtWebKit 
+    display.
+    """
+
+    def __init__(self, parent):
+        MediaPlayer.__init__(self, parent, u'webkit')
+        self.parent = parent
+        self.canBackground = True
+        self.audio_extensions_list = AUDIO_EXT
+        self.video_extensions_list = VIDEO_EXT
+
+    def get_media_display_css(self):
+        """
+        Add css style sheets to htmlbuilder
+        """
+        return VIDEO_CSS + FLASH_CSS
+
+    def get_media_display_javascript(self):
+        """
+        Add javascript functions to htmlbuilder
+        """
+        return VIDEO_JS + FLASH_JS
+
+    def get_media_display_html(self):
+        """
+        Add html code to htmlbuilder
+        """
+        return VIDEO_HTML + FLASH_HTML
+
+    def setup(self, display):
+        display.webView.resize(display.size())
+        display.webView.raise_()
+        self.hasOwnWidget = False
+
+    def check_available(self):
+        return True
+
+    def load(self, display):
+        log.debug(u'load vid in Webkit Controller')
+        controller = display.controller
+        if display.hasAudio:
+            volume = controller.media_info.volume
+            vol = float(volume) / float(100)
+        else:
+            vol = 0
+        path = controller.media_info.file_info.absoluteFilePath()
+        if controller.media_info.is_background:
+            loop = u'true'
+        else:
+            loop = u'false'
+        display.webView.setVisible(True)
+        if controller.media_info.file_info.suffix() == u'swf':
+            controller.media_info.is_flash = True
+            js = u'show_flash("load","%s");' % \
+                (path.replace(u'\\', u'\\\\'))
+        else:
+            js = u'show_video("init", "%s", %s, %s);' % \
+                (path.replace(u'\\', u'\\\\'), str(vol), loop)
+        display.frame.evaluateJavaScript(js)
+        return True
+
+    def resize(self, display):
+        controller = display.controller
+        display.webView.resize(display.size())
+
+    def play(self, display):
+        controller = display.controller
+        display.webLoaded = True
+        length = 0
+        self.set_visible(display, True)
+        if controller.media_info.is_flash:
+            display.frame.evaluateJavaScript(u'show_flash("play");')
+        else:
+            display.frame.evaluateJavaScript(u'show_video("play");')
+        # TODO add playing check and get the correct media length
+        controller.media_info.length = length
+        self.state = MediaState.Playing
+        display.webView.raise_()
+        return True
+
+    def pause(self, display):
+        controller = display.controller
+        if controller.media_info.is_flash:
+            display.frame.evaluateJavaScript(u'show_flash("pause");')
+        else:
+            display.frame.evaluateJavaScript(u'show_video("pause");')
+        self.state = MediaState.Paused
+
+    def stop(self, display):
+        controller = display.controller
+        if controller.media_info.is_flash:
+            display.frame.evaluateJavaScript(u'show_flash("stop");')
+        else:
+            display.frame.evaluateJavaScript(u'show_video("stop");')
+        controller.seekSlider.setSliderPosition(0)
+        self.state = MediaState.Stopped
+
+    def volume(self, display, vol):
+        controller = display.controller
+        # 1.0 is the highest value
+        if display.hasAudio:
+            vol = float(vol) / float(100)
+            if not controller.media_info.is_flash:
+                display.frame.evaluateJavaScript(
+                    u'show_video(null, null, %s);' % str(vol))
+
+    def seek(self, display, seekVal):
+        controller = display.controller
+        if controller.media_info.is_flash:
+            seek = seekVal
+            display.frame.evaluateJavaScript( \
+                u'show_flash("seek", null, null, "%s");' % (seek))
+        else:
+            seek = float(seekVal)/1000
+            display.frame.evaluateJavaScript( \
+                u'show_video("seek", null, null, null, "%f");' % (seek))
+
+    def reset(self, display):
+        controller = display.controller
+        if controller.media_info.is_flash:
+            display.frame.evaluateJavaScript(u'show_flash("close");')
+        else:
+            display.frame.evaluateJavaScript(u'show_video("close");')
+        self.state = MediaState.Off
+
+    def set_visible(self, display, status):
+        controller = display.controller
+        if status:
+            is_visible = "visible"
+        else:
+            is_visible = "hidden"
+        if controller.media_info.is_flash:
+            display.frame.evaluateJavaScript(u'show_flash( \
+                "setVisible", null, null, "%s");' % (is_visible))
+        else:
+            display.frame.evaluateJavaScript(u'show_video( \
+                "setVisible", null, null, null, "%s");' % (is_visible))
+
+    def update_ui(self, display):
+        controller = display.controller
+        if controller.media_info.is_flash:
+            currentTime = display.frame.evaluateJavaScript( \
+                u'show_flash("currentTime");').toInt()[0]
+            length = display.frame.evaluateJavaScript( \
+                u'show_flash("length");').toInt()[0]
+        else:
+            (currentTime, ok) = display.frame.evaluateJavaScript( \
+                u'show_video("currentTime");').toFloat()
+            # check if conversion was ok and value is not 'NaN'
+            if ok and currentTime != float('inf'):
+                currentTime = int(currentTime*1000)
+            (length, ok) = display.frame.evaluateJavaScript( \
+                u'show_video("length");').toFloat()
+            # check if conversion was ok and value is not 'NaN'
+            if ok and length != float('inf'):
+                length = int(length*1000)
+        if currentTime > 0:
+            controller.media_info.length = length
+            controller.seekSlider.setMaximum(length)
+            if not controller.seekSlider.isSliderDown():
+                controller.seekSlider.setSliderPosition(currentTime)

=== modified file 'openlp/core/ui/slidecontroller.py'
--- openlp/core/ui/slidecontroller.py	2011-11-28 18:03:38 +0000
+++ openlp/core/ui/slidecontroller.py	2011-11-30 20:32:27 +0000
@@ -34,9 +34,9 @@
 from PyQt4.phonon import Phonon
 
 from openlp.core.lib import OpenLPToolbar, Receiver, ItemCapabilities, \
-    translate, build_icon
+    translate, build_icon, ServiceItem, build_html, PluginManager, ServiceItem
 from openlp.core.lib.ui import UiStrings, shortcut_action
-from openlp.core.ui import HideMode, MainDisplay, ScreenList
+from openlp.core.ui import HideMode, MainDisplay, Display, ScreenList
 from openlp.core.utils.actions import ActionList, CategoryOrder
 
 log = logging.getLogger(__name__)
@@ -49,8 +49,29 @@
     def __init__(self, parent=None, name=None):
         QtGui.QTableWidget.__init__(self, parent.controller)
 
-
-class SlideController(QtGui.QWidget):
+class Controller(QtGui.QWidget):
+    """
+    Controller is a general controller widget.
+    """
+    def __init__(self, parent, isLive=False):
+        """
+        Set up the general Controller.
+        """
+        QtGui.QWidget.__init__(self, parent)
+        self.isLive = isLive
+        self.display = None
+
+    def sendToPlugins(self, *args):
+        """
+        This is the generic function to send signal for control widgets,
+        created from within other plugins
+        This function is needed to catch the current controller
+        """
+        sender = self.sender().objectName() if self.sender().objectName() else self.sender().text()
+        controller = self
+        Receiver.send_message('%s' % sender, [controller, args])
+
+class SlideController(Controller):
     """
     SlideController is the slide controller widget. This widget is what the
     user uses to control the displaying of verses/slides/etc on the screen.
@@ -59,13 +80,12 @@
         """
         Set up the Slide Controller.
         """
-        QtGui.QWidget.__init__(self, parent)
-        self.isLive = isLive
-        self.display = None
+        Controller.__init__(self, parent, isLive)
         self.screens = ScreenList.get_instance()
         self.ratio = float(self.screens.current[u'size'].width()) / \
             float(self.screens.current[u'size'].height())
         self.imageManager = self.parent().imageManager
+        self.mediaController = self.parent().mediaController
         self.loopList = [
             u'Play Slides Menu',
             u'Loop Separator',
@@ -74,7 +94,10 @@
         self.songEditList = [
             u'Edit Song',
         ]
-        self.volume = 10
+        self.nextPreviousList = [
+            u'Previous Slide', 
+            u'Next Slide'
+        ]
         self.timer_id = 0
         self.songEdit = False
         self.selectedRow = 0
@@ -140,14 +163,14 @@
             self.toolbar.sizePolicy().hasHeightForWidth())
         self.toolbar.setSizePolicy(sizeToolbarPolicy)
         self.previousItem = self.toolbar.addToolbarButton(
-            translate('OpenLP.SlideController', 'Previous Slide'),
+            u'Previous Slide',
             u':/slides/slide_previous.png',
             translate('OpenLP.SlideController', 'Move to previous.'),
             self.onSlideSelectedPrevious,
             shortcuts=[QtCore.Qt.Key_Up, QtCore.Qt.Key_PageUp],
             context=QtCore.Qt.WidgetWithChildrenShortcut)
         self.nextItem = self.toolbar.addToolbarButton(
-            translate('OpenLP.SlideController', 'Next Slide'),
+            u'Next Slide',
             u':/slides/slide_next.png',
             translate('OpenLP.SlideController', 'Move to next.'),
             self.onSlideSelectedNext,
@@ -234,20 +257,8 @@
                 'Edit and reload song preview.'),
                 self.onEditSong)
         self.controllerLayout.addWidget(self.toolbar)
-        # Build a Media ToolBar
-        self.mediabar = OpenLPToolbar(self)
-        self.mediabar.addToolbarButton(
-            u'Media Start', u':/slides/media_playback_start.png',
-            translate('OpenLP.SlideController', 'Start playing media.'),
-            self.onMediaPlay)
-        self.mediabar.addToolbarButton(
-            u'Media Pause', u':/slides/media_playback_pause.png',
-            translate('OpenLP.SlideController', 'Start playing media.'),
-            self.onMediaPause)
-        self.mediabar.addToolbarButton(
-            u'Media Stop', u':/slides/media_playback_stop.png',
-            translate('OpenLP.SlideController', 'Start playing media.'),
-            self.onMediaStop)
+        # Build the Media Toolbar
+        self.mediaController.add_controller_items(self, self.controllerLayout)
         if self.isLive:
             # Build the Song Toolbar
             self.songMenu = QtGui.QToolButton(self.toolbar)
@@ -263,23 +274,6 @@
                 translate('OpenLP.SlideController', 'Pause audio.'),
                 self.onAudioPauseClicked, True)
             self.audioPauseItem.setVisible(False)
-            # Build the volumeSlider.
-            self.volumeSlider = QtGui.QSlider(QtCore.Qt.Horizontal)
-            self.volumeSlider.setTickInterval(1)
-            self.volumeSlider.setTickPosition(QtGui.QSlider.TicksAbove)
-            self.volumeSlider.setMinimum(0)
-            self.volumeSlider.setMaximum(10)
-        else:
-            # Build the seekSlider.
-            self.seekSlider = Phonon.SeekSlider()
-            self.seekSlider.setGeometry(QtCore.QRect(90, 260, 221, 24))
-            self.seekSlider.setObjectName(u'seekSlider')
-            self.mediabar.addToolbarWidget(u'Seek Slider', self.seekSlider)
-            self.volumeSlider = Phonon.VolumeSlider()
-        self.volumeSlider.setGeometry(QtCore.QRect(90, 260, 221, 24))
-        self.volumeSlider.setObjectName(u'volumeSlider')
-        self.mediabar.addToolbarWidget(u'Audio Volume', self.volumeSlider)
-        self.controllerLayout.addWidget(self.mediabar)
         # Screen preview area
         self.previewFrame = QtGui.QFrame(self.splitter)
         self.previewFrame.setGeometry(QtCore.QRect(0, 0, 300, 300 * self.ratio))
@@ -296,17 +290,13 @@
         self.slideLayout = QtGui.QVBoxLayout()
         self.slideLayout.setSpacing(0)
         self.slideLayout.setMargin(0)
-        self.slideLayout.setObjectName(u'slideLayout')
-        if not self.isLive:
-            self.mediaObject = Phonon.MediaObject(self)
-            self.video = Phonon.VideoWidget()
-            self.video.setVisible(False)
-            self.audio = Phonon.AudioOutput(Phonon.VideoCategory,
-                self.mediaObject)
-            Phonon.createPath(self.mediaObject, self.video)
-            Phonon.createPath(self.mediaObject, self.audio)
-            self.video.setGeometry(QtCore.QRect(0, 0, 300, 225))
-            self.slideLayout.insertWidget(0, self.video)
+        self.slideLayout.setObjectName(u'SlideLayout')
+        self.previewDisplay = Display(self, self.isLive, self)
+        self.previewDisplay.setGeometry(QtCore.QRect(0, 0, 300, 300))
+        self.previewDisplay.screen = {u'size':self.previewDisplay.geometry()}
+        self.previewDisplay.setup()
+        self.slideLayout.insertWidget(0, self.previewDisplay)
+        self.previewDisplay.hide()
         # Actual preview screen
         self.slidePreview = QtGui.QLabel(self)
         sizePolicy = QtGui.QSizePolicy(QtGui.QSizePolicy.Fixed,
@@ -415,8 +405,6 @@
         QtCore.QObject.connect(self.previewListWidget,
             QtCore.SIGNAL(u'clicked(QModelIndex)'), self.onSlideSelected)
         if self.isLive:
-            QtCore.QObject.connect(self.volumeSlider,
-                QtCore.SIGNAL(u'sliderReleased()'), self.mediaVolume)
             QtCore.QObject.connect(Receiver.get_receiver(),
                 QtCore.SIGNAL(u'slidecontroller_live_spin_delay'),
                 self.receiveSpinDelay)
@@ -426,7 +414,6 @@
                 QtCore.SIGNAL(u'doubleClicked(QModelIndex)'),
                 self.onGoLiveClick)
             self.toolbar.makeWidgetsInvisible(self.songEditList)
-        self.mediabar.setVisible(False)
         if self.isLive:
             self.setLiveHotkeys(self)
             self.__addActionsToWidget(self.previewListWidget)
@@ -577,7 +564,7 @@
 
     def liveEscape(self):
         self.display.setVisible(False)
-        self.display.videoStop()
+        self.mediaController.video_stop([self])
 
     def servicePrevious(self):
         """
@@ -617,14 +604,22 @@
         # rebuild display as screen size changed
         if self.display:
             self.display.close()
-        self.display = MainDisplay(self, self.imageManager, self.isLive)
+        self.display = MainDisplay(self, self.imageManager, self.isLive,
+            self)
         self.display.setup()
         if self.isLive:
             self.__addActionsToWidget(self.display)
         # The SlidePreview's ratio.
         self.ratio = float(self.screens.current[u'size'].width()) / \
             float(self.screens.current[u'size'].height())
+        self.mediaController.setup_display(self.display)
         self.previewSizeChanged()
+        self.previewDisplay.setup()
+        serviceItem = ServiceItem()
+        self.previewDisplay.webView.setHtml(build_html(serviceItem, 
+            self.previewDisplay.screen, None, self.isLive, None, 
+            plugins=PluginManager.get_instance().plugins))
+        self.mediaController.setup_display(self.previewDisplay)
         if self.serviceItem:
             self.refreshServiceItem()
 
@@ -646,11 +641,17 @@
             max_height = self.previewFrame.height() - self.grid.margin() * 2
             self.slidePreview.setFixedSize(QtCore.QSize(max_height * self.ratio,
                 max_height))
+            self.previewDisplay.setFixedSize(QtCore.QSize(max_height * self.ratio,
+                max_height))
+            self.previewDisplay.screen = {u'size':self.previewDisplay.geometry()}
         else:
             # We have to take the width as limit.
             max_width = self.previewFrame.width() - self.grid.margin() * 2
             self.slidePreview.setFixedSize(QtCore.QSize(max_width,
                 max_width / self.ratio))
+            self.previewDisplay.setFixedSize(QtCore.QSize(max_width,
+                max_width / self.ratio))
+            self.previewDisplay.screen = {u'size':self.previewDisplay.geometry()}
         # Make sure that the frames have the correct size.
         self.previewListWidget.setColumnWidth(0,
             self.previewListWidget.viewport().size().width())
@@ -711,12 +712,13 @@
             len(item.get_frames()) > 1:
             self.toolbar.makeWidgetsVisible(self.loopList)
         if item.is_media():
-            self.toolbar.setVisible(False)
             self.mediabar.setVisible(True)
+            self.toolbar.makeWidgetsInvisible(self.nextPreviousList)
         else:
             # Work-around for OS X, hide and then show the toolbar
             # See bug #791050
-            self.toolbar.show()
+            self.toolbar.makeWidgetsVisible(self.nextPreviousList)
+        self.toolbar.show()
 
     def enablePreviewToolBar(self, item):
         """
@@ -730,13 +732,13 @@
         if item.is_capable(ItemCapabilities.CanEdit) and item.from_plugin:
             self.toolbar.makeWidgetsVisible(self.songEditList)
         elif item.is_media():
-            self.toolbar.setVisible(False)
             self.mediabar.setVisible(True)
-            self.volumeSlider.setAudioOutput(self.audio)
+            self.toolbar.makeWidgetsInvisible(self.nextPreviousList)
         if not item.is_media():
             # Work-around for OS X, hide and then show the toolbar
             # See bug #791050
-            self.toolbar.show()
+            self.toolbar.makeWidgetsVisible(self.nextPreviousList)
+        self.toolbar.show()
 
     def refreshServiceItem(self):
         """
@@ -1325,72 +1327,18 @@
         """
         log.debug(u'SlideController onMediaStart')
         file = os.path.join(item.get_frame_path(), item.get_frame_title())
-        if self.isLive:
-            self.display.video(file, self.volume)
-            self.volumeSlider.setValue(self.volume)
-        else:
-            self.mediaObject.stop()
-            self.mediaObject.clearQueue()
-            self.mediaObject.setCurrentSource(Phonon.MediaSource(file))
-            self.seekSlider.setMediaObject(self.mediaObject)
-            self.seekSlider.show()
-            self.onMediaPlay()
-
-    def mediaVolume(self):
-        """
-        Respond to the release of Volume Slider
-        """
-        log.debug(u'SlideController mediaVolume')
-        self.volume = self.volumeSlider.value()
-        self.display.videoVolume(self.volume)
-
-    def onMediaPause(self):
-        """
-        Respond to the Pause from the media Toolbar
-        """
-        log.debug(u'SlideController onMediaPause')
-        if self.isLive:
-            self.display.videoPause()
-        else:
-            self.mediaObject.pause()
-
-    def onMediaPlay(self):
-        """
-        Respond to the Play from the media Toolbar
-        """
-        log.debug(u'SlideController onMediaPlay')
-        if self.isLive:
-            self.display.videoPlay()
-        else:
+        self.mediaController.video(self, file, False, False)
+        if not self.isLive or self.mediaController.withLivePreview:
+            self.previewDisplay.show()
             self.slidePreview.hide()
-            self.video.show()
-            self.mediaObject.play()
-
-    def onMediaStop(self):
-        """
-        Respond to the Stop from the media Toolbar
-        """
-        log.debug(u'SlideController onMediaStop')
-        if self.isLive:
-            self.display.videoStop()
-        else:
-            self.mediaObject.stop()
-            self.video.hide()
-        self.slidePreview.clear()
-        self.slidePreview.show()
 
     def onMediaClose(self):
         """
         Respond to a request to close the Video
         """
-        log.debug(u'SlideController onMediaStop')
-        if self.isLive:
-            self.display.resetVideo()
-        else:
-            self.mediaObject.stop()
-            self.mediaObject.clearQueue()
-            self.video.hide()
-        self.slidePreview.clear()
+        log.debug(u'SlideController onMediaClose')
+        self.mediaController.video_reset(self)
+        self.previewDisplay.hide()
         self.slidePreview.show()
 
     def _resetBlank(self):

=== modified file 'openlp/plugins/media/lib/mediaitem.py'
--- openlp/plugins/media/lib/mediaitem.py	2011-10-15 11:30:39 +0000
+++ openlp/plugins/media/lib/mediaitem.py	2011-11-30 20:32:27 +0000
@@ -25,7 +25,6 @@
 # Temple Place, Suite 330, Boston, MA 02111-1307 USA                          #
 ###############################################################################
 
-from datetime import datetime
 import logging
 import os
 import locale
@@ -34,12 +33,17 @@
 from PyQt4.phonon import Phonon
 
 from openlp.core.lib import MediaManagerItem, build_icon, ItemCapabilities, \
-    SettingsManager, translate, check_item_selected, Receiver, MediaType
-from openlp.core.lib.ui import UiStrings, critical_error_message_box
+    SettingsManager, translate, check_item_selected, Receiver, MediaType, \
+    ServiceItem, build_html
+from openlp.core.lib.ui import UiStrings, critical_error_message_box, \
+    media_item_combo_box
+from openlp.core.ui import Controller, Display
 
 log = logging.getLogger(__name__)
 
 CLAPPERBOARD = QtGui.QImage(u':/media/media_video.png')
+#TODO: Add an appropriate Icon for DVDs, CDs, ...
+DVD_ICON = QtGui.QImage(u':/media/media_video.png')
 
 class MediaMediaItem(MediaManagerItem):
     """
@@ -51,16 +55,38 @@
         self.iconPath = u'images/image'
         self.background = False
         self.previewFunction = CLAPPERBOARD
+        self.Automatic = u''
         MediaManagerItem.__init__(self, parent, plugin, icon)
         self.singleServiceItem = False
         self.hasSearch = True
         self.mediaObject = None
+        self.mediaController = Controller(parent)
+        self.mediaController.controllerLayout = QtGui.QVBoxLayout()
+        self.plugin.mediaController.add_controller_items(self.mediaController, \
+            self.mediaController.controllerLayout)
+        self.plugin.mediaController.set_controls_visible(self.mediaController, \
+            False)
+        self.mediaController.previewDisplay = Display(self.mediaController, \
+            False, self.mediaController)
+        self.mediaController.previewDisplay.setGeometry(
+            QtCore.QRect(0, 0, 300, 300))
+        self.mediaController.previewDisplay.screen = \
+            {u'size':self.mediaController.previewDisplay.geometry()}
+        self.mediaController.previewDisplay.setup()
+        serviceItem = ServiceItem()
+        self.mediaController.previewDisplay.webView.setHtml(build_html( \
+            serviceItem, self.mediaController.previewDisplay.screen, None, \
+            False, None))
+        self.mediaController.previewDisplay.setup()
+        self.plugin.mediaController.setup_display( \
+            self.mediaController.previewDisplay)
+        self.mediaController.previewDisplay.hide()
+
         QtCore.QObject.connect(Receiver.get_receiver(),
             QtCore.SIGNAL(u'video_background_replaced'),
             self.videobackgroundReplaced)
         QtCore.QObject.connect(Receiver.get_receiver(),
-            QtCore.SIGNAL(u'openlp_phonon_creation'),
-            self.createPhonon)
+            QtCore.SIGNAL(u'mediaitem_media_rebuild'), self.rebuild)
         # Allow DnD from the desktop
         self.listView.activateDnD()
 
@@ -74,6 +100,10 @@
         self.replaceAction.setToolTip(UiStrings().ReplaceLiveBG)
         self.resetAction.setText(UiStrings().ResetBG)
         self.resetAction.setToolTip(UiStrings().ResetLiveBG)
+        self.Automatic = translate('MediaPlugin.MediaItem',
+            'Automatic')
+        self.displayTypeLabel.setText(
+            translate('MediaPlugin.MediaItem', 'Use Player:'))
 
     def requiredIcons(self):
         MediaManagerItem.requiredIcons(self)
@@ -92,13 +122,34 @@
         self.resetAction = self.addToolbarButton(u'', u'',
             u':/system/system_close.png', self.onResetClick, False)
         self.resetAction.setVisible(False)
+        self.mediaWidget = QtGui.QWidget(self)
+        self.mediaWidget.setObjectName(u'mediaWidget')
+        self.displayLayout = QtGui.QFormLayout(self.mediaWidget)
+        self.displayLayout.setMargin(self.displayLayout.spacing())
+        self.displayLayout.setObjectName(u'displayLayout')
+        self.displayTypeLabel = QtGui.QLabel(self.mediaWidget)
+        self.displayTypeLabel.setObjectName(u'displayTypeLabel')
+        self.displayTypeComboBox = media_item_combo_box(
+            self.mediaWidget, u'displayTypeComboBox')
+        self.displayTypeLabel.setBuddy(self.displayTypeComboBox)
+        self.displayLayout.addRow(self.displayTypeLabel,
+            self.displayTypeComboBox)
+        # Add the Media widget to the page layout
+        self.pageLayout.addWidget(self.mediaWidget)
+        QtCore.QObject.connect(self.displayTypeComboBox,
+            QtCore.SIGNAL(u'currentIndexChanged (int)'), self.overridePlayerChanged)
+
+    def overridePlayerChanged(self, index):
+        Receiver.send_message(u'media_override_player', \
+            u'%s' % self.displayTypeComboBox.currentText())
 
     def onResetClick(self):
         """
-        Called to reset the Live backgound with the media selected.
+        Called to reset the Live background with the media selected,
         """
+        self.plugin.liveController.mediaController.video_reset( \
+            self.plugin.liveController)
         self.resetAction.setVisible(False)
-        self.plugin.liveController.display.resetVideo()
 
     def videobackgroundReplaced(self):
         """
@@ -108,7 +159,7 @@
 
     def onReplaceClick(self):
         """
-        Called to replace Live backgound with the media selected.
+        Called to replace Live background with the media selected.
         """
         if check_item_selected(self.listView,
             translate('MediaPlugin.MediaItem',
@@ -116,8 +167,8 @@
             item = self.listView.currentItem()
             filename = unicode(item.data(QtCore.Qt.UserRole).toString())
             if os.path.exists(filename):
-                (path, name) = os.path.split(filename)
-                if self.plugin.liveController.display.video(filename, 0, True):
+                if self.plugin.liveController.mediaController.video( \
+                    self.plugin.liveController, filename, True, True):
                     self.resetAction.setVisible(True)
                 else:
                     critical_error_message_box(UiStrings().LiveBGError,
@@ -144,30 +195,18 @@
                         unicode(translate('MediaPlugin.MediaItem',
                             'The file %s no longer exists.')) % filename)
             return False
-        self.mediaObject.stop()
-        self.mediaObject.clearQueue()
-        self.mediaObject.setCurrentSource(Phonon.MediaSource(filename))
-        if not self.mediaStateWait(Phonon.StoppedState):
-            critical_error_message_box(UiStrings().UnsupportedFile,
-                    UiStrings().UnsupportedFile)
+        self.mediaLength = 0
+        if self.plugin.mediaController.video( \
+                    self.mediaController, filename, False, False):
+            self.mediaLength = self.mediaController.media_info.length
+            service_item.media_length = self.mediaLength
+            self.plugin.mediaController.video_reset(self.mediaController)
+            if self.mediaLength > 0:
+                service_item.add_capability(
+                    ItemCapabilities.HasVariableStartTime)
+        else:
             return False
-        # File too big for processing
-        if os.path.getsize(filename) <= 52428800: # 50MiB
-            self.mediaObject.play()
-            if not self.mediaStateWait(Phonon.PlayingState) \
-                or self.mediaObject.currentSource().type() \
-                == Phonon.MediaSource.Invalid:
-                self.mediaObject.stop()
-                critical_error_message_box(
-                    translate('MediaPlugin.MediaItem', 'File Too Big'),
-                    translate('MediaPlugin.MediaItem', 'The file you are '
-                        'trying to load is too big. Please reduce it to less '
-                        'than 50MiB.'))
-                return False
-            self.mediaObject.stop()
-            service_item.media_length = self.mediaObject.totalTime() / 1000
-            service_item.add_capability(
-                ItemCapabilities.HasVariableStartTime)
+        service_item.media_length = self.mediaLength
         service_item.title = unicode(self.plugin.nameStrings[u'singular'])
         service_item.add_capability(ItemCapabilities.RequiresMedia)
         # force a non-existent theme
@@ -177,23 +216,45 @@
         service_item.add_from_command(path, name, frame)
         return True
 
-    def mediaStateWait(self, mediaState):
-        """
-        Wait for the video to change its state. Wait no longer than 5 seconds.
-        """
-        start = datetime.now()
-        while self.mediaObject.state() != mediaState:
-            if self.mediaObject.state() == Phonon.ErrorState:
-                return False
-            Receiver.send_message(u'openlp_process_events')
-            if (datetime.now() - start).seconds > 5:
-                return False
-        return True
-
     def initialise(self):
         self.listView.clear()
         self.listView.setIconSize(QtCore.QSize(88, 50))
         self.loadList(SettingsManager.load_list(self.settingsSection, u'media'))
+        self.populateDisplayTypes()
+
+    def rebuild(self):
+        """
+        Rebuild the tab in the media manager when changes are made in
+        the settings
+        """
+        self.populateDisplayTypes()
+        self.onNewFileMasks = unicode(translate('MediaPlugin.MediaItem',
+            'Videos (%s);;Audio (%s);;%s (*)')) % (
+            u' '.join(self.plugin.video_extensions_list),
+            u' '.join(self.plugin.audio_extensions_list), UiStrings().AllFiles)
+        
+
+    def populateDisplayTypes(self):
+        """
+        Load the combobox with the enabled media players,
+        allowing user to select a specific player if settings allow
+        """
+        self.displayTypeComboBox.clear()
+        playerSettings = str(QtCore.QSettings().value(u'media/players',
+            QtCore.QVariant(u'webkit')).toString())
+        usedPlayers = playerSettings.split(u',')
+        for title in usedPlayers:
+            # load the drop down selection
+            self.displayTypeComboBox.addItem(title)
+        if self.displayTypeComboBox.count() > 1:
+            self.displayTypeComboBox.insertItem(0, self.Automatic)
+            self.displayTypeComboBox.setCurrentIndex(0)
+        if QtCore.QSettings().value(self.settingsSection + u'/override player',
+            QtCore.QVariant(QtCore.Qt.Unchecked)) == QtCore.Qt.Checked:
+            self.mediaWidget.show()
+        else:
+            self.mediaWidget.hide()
+
 
     def onDeleteClick(self):
         """
@@ -214,10 +275,18 @@
         media.sort(cmp=locale.strcoll,
             key=lambda filename: os.path.split(unicode(filename))[1].lower())
         for track in media:
-            filename = os.path.split(unicode(track))[1]
-            item_name = QtGui.QListWidgetItem(filename)
-            item_name.setIcon(build_icon(CLAPPERBOARD))
-            item_name.setData(QtCore.Qt.UserRole, QtCore.QVariant(track))
+            track_info = QtCore.QFileInfo(track)
+            if not track_info.isFile():
+                filename = os.path.split(unicode(track))[1]
+                item_name = QtGui.QListWidgetItem(filename)
+                item_name.setIcon(build_icon(CLAPPERBOARD))
+                item_name.setData(QtCore.Qt.UserRole, QtCore.QVariant(track))
+            else:
+                filename = os.path.split(unicode(track))[1]
+                item_name = QtGui.QListWidgetItem(filename)
+                #TODO: add the appropriate Icon
+                #item_name.setIcon(build_icon(DVD_ICON))
+                item_name.setData(QtCore.Qt.UserRole, QtCore.QVariant(track))
             item_name.setToolTip(track)
             self.listView.addItem(item_name)
 
@@ -234,11 +303,6 @@
         media = filter(lambda x: os.path.splitext(x)[1] in ext, media)
         return media
 
-    def createPhonon(self):
-        log.debug(u'CreatePhonon')
-        if not self.mediaObject:
-            self.mediaObject = Phonon.MediaObject(self)
-
     def search(self, string):
         files = SettingsManager.load_list(self.settingsSection, u'media')
         results = []

=== modified file 'openlp/plugins/media/lib/mediatab.py'
--- openlp/plugins/media/lib/mediatab.py	2011-06-12 16:02:52 +0000
+++ openlp/plugins/media/lib/mediatab.py	2011-11-30 20:32:27 +0000
@@ -28,51 +28,181 @@
 from PyQt4 import QtCore, QtGui
 
 from openlp.core.lib import SettingsTab, translate, Receiver
+from openlp.core.lib.ui import UiStrings, critical_error_message_box
 
 class MediaTab(SettingsTab):
     """
     MediaTab is the Media settings tab in the settings dialog.
     """
-    def __init__(self, parent, title, visible_title, icon_path):
+    def __init__(self, parent, title, visible_title, media_players, icon_path):
+        self.media_players = media_players
         SettingsTab.__init__(self, parent, title, visible_title, icon_path)
 
     def setupUi(self):
         self.setObjectName(u'MediaTab')
         SettingsTab.setupUi(self)
-        self.mediaModeGroupBox = QtGui.QGroupBox(self.leftColumn)
-        self.mediaModeGroupBox.setObjectName(u'mediaModeGroupBox')
-        self.mediaModeLayout = QtGui.QFormLayout(self.mediaModeGroupBox)
-        self.mediaModeLayout.setObjectName(u'mediaModeLayout')
-        self.usePhononCheckBox = QtGui.QCheckBox(self.mediaModeGroupBox)
-        self.usePhononCheckBox.setObjectName(u'usePhononCheckBox')
-        self.mediaModeLayout.addRow(self.usePhononCheckBox)
-        self.leftLayout.addWidget(self.mediaModeGroupBox)
+        self.mediaPlayerGroupBox = QtGui.QGroupBox(self.leftColumn)
+        self.mediaPlayerGroupBox.setObjectName(u'mediaPlayerGroupBox')
+        self.mediaPlayerLayout = QtGui.QVBoxLayout(self.mediaPlayerGroupBox)
+        self.mediaPlayerLayout.setObjectName(u'mediaPlayerLayout')
+        self.PlayerCheckBoxes = {}
+        for key in self.media_players:
+            player = self.media_players[key]
+            checkbox = QtGui.QCheckBox(self.mediaPlayerGroupBox)
+            checkbox.setEnabled(player.available)
+            checkbox.setObjectName(player.name + u'CheckBox')
+            self.PlayerCheckBoxes[player.name] = checkbox
+            self.mediaPlayerLayout.addWidget(checkbox)
+        self.leftLayout.addWidget(self.mediaPlayerGroupBox)
+        self.playerOrderGroupBox = QtGui.QGroupBox(self.leftColumn)
+        self.playerOrderGroupBox.setObjectName(u'playerOrderGroupBox')
+        self.playerOrderLayout = QtGui.QVBoxLayout(self.playerOrderGroupBox)
+        self.playerOrderLayout.setObjectName(u'playerOrderLayout')
+        self.playerOrderlistWidget = QtGui.QListWidget( \
+            self.playerOrderGroupBox)
+        sizePolicy = QtGui.QSizePolicy(QtGui.QSizePolicy.Minimum,
+            QtGui.QSizePolicy.Expanding)
+        sizePolicy.setHorizontalStretch(0)
+        sizePolicy.setVerticalStretch(0)
+        sizePolicy.setHeightForWidth(self.playerOrderlistWidget. \
+            sizePolicy().hasHeightForWidth())
+        self.playerOrderlistWidget.setSizePolicy(sizePolicy)
+        self.playerOrderlistWidget.setVerticalScrollBarPolicy( \
+            QtCore.Qt.ScrollBarAsNeeded)
+        self.playerOrderlistWidget.setHorizontalScrollBarPolicy( \
+            QtCore.Qt.ScrollBarAlwaysOff)
+        self.playerOrderlistWidget.setEditTriggers( \
+            QtGui.QAbstractItemView.NoEditTriggers)
+        self.playerOrderlistWidget.setObjectName(u'playerOrderlistWidget')
+        self.playerOrderLayout.addWidget(self.playerOrderlistWidget)
+        self.orderingButtonsWidget = QtGui.QWidget(self.playerOrderGroupBox)
+        self.orderingButtonsWidget.setObjectName(u'orderingButtonsWidget')
+        self.orderingButtonLayout = QtGui.QHBoxLayout( \
+            self.orderingButtonsWidget)
+        self.orderingButtonLayout.setObjectName(u'orderingButtonLayout')
+        self.orderingDownButton = QtGui.QPushButton(self.orderingButtonsWidget)
+        self.orderingDownButton.setObjectName(u'orderingDownButton')
+        self.orderingButtonLayout.addWidget(self.orderingDownButton)
+        self.orderingUpButton = QtGui.QPushButton(self.playerOrderGroupBox)
+        self.orderingUpButton.setObjectName(u'orderingUpButton')
+        self.orderingButtonLayout.addWidget(self.orderingUpButton)
+        self.playerOrderLayout.addWidget(self.orderingButtonsWidget)
+        self.leftLayout.addWidget(self.playerOrderGroupBox)
+        self.AdvancedGroupBox = QtGui.QGroupBox(self.leftColumn)
+        self.AdvancedGroupBox.setObjectName(u'AdvancedGroupBox')
+        self.AdvancedLayout = QtGui.QVBoxLayout(self.AdvancedGroupBox)
+        self.AdvancedLayout.setObjectName(u'AdvancedLayout')
+        self.OverridePlayerCheckBox = QtGui.QCheckBox(self.AdvancedGroupBox)
+        self.OverridePlayerCheckBox.setObjectName(u'OverridePlayerCheckBox')
+        self.AdvancedLayout.addWidget(self.OverridePlayerCheckBox)
+        self.leftLayout.addWidget(self.AdvancedGroupBox)
         self.leftLayout.addStretch()
         self.rightLayout.addStretch()
-        QtCore.QObject.connect(self.usePhononCheckBox,
-            QtCore.SIGNAL(u'stateChanged(int)'),
-            self.onUsePhononCheckBoxChanged)
+        for key in self.media_players:
+            player = self.media_players[key]
+            checkbox = self.PlayerCheckBoxes[player.name]
+            QtCore.QObject.connect(checkbox,
+                QtCore.SIGNAL(u'stateChanged(int)'),
+                self.onPlayerCheckBoxChanged)
+        QtCore.QObject.connect(self.orderingUpButton,
+            QtCore.SIGNAL(u'pressed()'), self.onOrderingUpButtonPressed)
+        QtCore.QObject.connect(self.orderingDownButton,
+            QtCore.SIGNAL(u'pressed()'), self.onOrderingDownButtonPressed)
 
     def retranslateUi(self):
-        self.mediaModeGroupBox.setTitle(
-            translate('MediaPlugin.MediaTab', 'Media Display'))
-        self.usePhononCheckBox.setText(
-            translate('MediaPlugin.MediaTab', 'Use Phonon for video playback'))
-
-    def onUsePhononCheckBoxChanged(self, check_state):
-        self.usePhonon = (check_state == QtCore.Qt.Checked)
-        self.usePhononChanged = True
+        self.mediaPlayerGroupBox.setTitle(
+            translate('MediaPlugin.MediaTab', 'Available Media Players'))
+        for key in self.media_players:
+            player = self.media_players[key]
+            checkbox = self.PlayerCheckBoxes[player.name]
+            if player.available:
+                checkbox.setText(player.name)
+            else:
+                checkbox.setText(
+                    unicode(translate('MediaPlugin.MediaTab',
+                    '%s (unavailable)')) % player.name)
+        self.playerOrderGroupBox.setTitle(
+            translate('MediaPlugin.MediaTab', 'Player Order'))
+        self.orderingDownButton.setText(
+            translate('MediaPlugin.MediaTab', 'Down'))
+        self.orderingUpButton.setText(
+            translate('MediaPlugin.MediaTab', 'Up'))
+        self.AdvancedGroupBox.setTitle(UiStrings().Advanced)
+        self.OverridePlayerCheckBox.setText(
+            translate('MediaPlugin.MediaTab',
+            'Allow media player to be overriden'))
+
+    def onPlayerCheckBoxChanged(self, check_state):
+        player = self.sender().text()
+        if check_state == QtCore.Qt.Checked:
+            if player not in self.usedPlayers:
+                self.usedPlayers.append(player)
+        else:
+            self.usedPlayers.takeAt(self.usedPlayers.indexOf(player))
+        self.updatePlayerList()
+
+    def updatePlayerList(self):
+        self.playerOrderlistWidget.clear()
+        for player in self.usedPlayers:
+            if player in self.PlayerCheckBoxes.keys():
+                if len(self.usedPlayers) == 1:
+                    # at least one media player have to stay active
+                    self.PlayerCheckBoxes[u'%s' % player].setEnabled(False)
+                else:
+                    self.PlayerCheckBoxes[u'%s' % player].setEnabled(True)
+                self.playerOrderlistWidget.addItem(player)
+
+    def onOrderingUpButtonPressed(self):
+        currentRow = self.playerOrderlistWidget.currentRow()
+        if currentRow > 0:
+            item = self.playerOrderlistWidget.takeItem(currentRow)
+            self.playerOrderlistWidget.insertItem(currentRow - 1, item)
+            self.playerOrderlistWidget.setCurrentRow(currentRow - 1)
+            self.usedPlayers.move(currentRow, currentRow - 1)
+
+    def onOrderingDownButtonPressed(self):
+        currentRow = self.playerOrderlistWidget.currentRow()
+        if currentRow < self.playerOrderlistWidget.count() - 1:
+            item = self.playerOrderlistWidget.takeItem(currentRow)
+            self.playerOrderlistWidget.insertItem(currentRow + 1, item)
+            self.playerOrderlistWidget.setCurrentRow(currentRow + 1)
+            self.usedPlayers.move(currentRow, currentRow + 1)
 
     def load(self):
-        self.usePhonon = QtCore.QSettings().value(
-            self.settingsSection + u'/use phonon',
-            QtCore.QVariant(True)).toBool()
-        self.usePhononCheckBox.setChecked(self.usePhonon)
+        self.usedPlayers = QtCore.QSettings().value(
+            self.settingsSection + u'/players',
+            QtCore.QVariant(u'webkit')).toString().split(u',')
+        for key in self.media_players:
+            player = self.media_players[key]
+            checkbox = self.PlayerCheckBoxes[player.name]
+            if player.available and player.name in self.usedPlayers:
+                checkbox.setChecked(True)
+        self.updatePlayerList()
+        self.OverridePlayerCheckBox.setChecked(QtCore.QSettings().value(
+            self.settingsSection + u'/override player',
+            QtCore.QVariant(QtCore.Qt.Unchecked)).toInt()[0])
 
     def save(self):
-        oldUsePhonon = QtCore.QSettings().value(
-            u'media/use phonon', QtCore.QVariant(True)).toBool()
-        if oldUsePhonon != self.usePhonon:
-            QtCore.QSettings().setValue(self.settingsSection + u'/use phonon',
-                QtCore.QVariant(self.usePhonon))
+        override_changed = False
+        player_string_changed = False
+        oldPlayerString = QtCore.QSettings().value(
+            self.settingsSection + u'/players',
+            QtCore.QVariant(u'webkit')).toString()
+        newPlayerString = self.usedPlayers.join(u',')
+        if oldPlayerString != newPlayerString:
+            # clean old Media stuff
+            QtCore.QSettings().setValue(self.settingsSection + u'/players',
+                QtCore.QVariant(newPlayerString))
+            player_string_changed = True
+            override_changed = True
+        setting_key = self.settingsSection + u'/override player'
+        if QtCore.QSettings().value(setting_key) != \
+            self.OverridePlayerCheckBox.checkState():
+            QtCore.QSettings().setValue(setting_key,
+                QtCore.QVariant(self.OverridePlayerCheckBox.checkState()))
+            override_changed = True
+        if override_changed:
+            Receiver.send_message(u'mediaitem_media_rebuild')
+        if player_string_changed:
+            Receiver.send_message(u'mediaitem_media_rebuild')
             Receiver.send_message(u'config_screen_changed')

=== modified file 'openlp/plugins/media/mediaplugin.py'
--- openlp/plugins/media/mediaplugin.py	2011-07-23 21:29:24 +0000
+++ openlp/plugins/media/mediaplugin.py	2011-11-30 20:32:27 +0000
@@ -26,9 +26,7 @@
 ###############################################################################
 
 import logging
-import mimetypes
-
-from PyQt4.phonon import Phonon
+import os
 
 from openlp.core.lib import Plugin, StringContent, build_icon, translate
 from openlp.plugins.media.lib import MediaMediaItem, MediaTab
@@ -40,57 +38,28 @@
 
     def __init__(self, plugin_helpers):
         Plugin.__init__(self, u'media', plugin_helpers,
-            MediaMediaItem, MediaTab)
+            MediaMediaItem)
         self.weight = -6
         self.icon_path = u':/plugins/plugin_media.png'
         self.icon = build_icon(self.icon_path)
         # passed with drag and drop messages
         self.dnd_id = u'Media'
-        self.additional_extensions = {
-            u'audio/ac3': [u'.ac3'],
-            u'audio/flac': [u'.flac'],
-            u'audio/x-m4a': [u'.m4a'],
-            u'audio/midi': [u'.mid', u'.midi'],
-            u'audio/x-mp3': [u'.mp3'],
-            u'audio/mpeg': [u'.mp3', u'.mp2', u'.mpga', u'.mpega', u'.m4a'],
-            u'audio/qcelp': [u'.qcp'],
-            u'audio/x-wma': [u'.wma'],
-            u'audio/x-ms-wma': [u'.wma'],
-            u'video/x-flv': [u'.flv'],
-            u'video/x-matroska': [u'.mpv', u'.mkv'],
-            u'video/x-wmv': [u'.wmv'],
-            u'video/x-ms-wmv': [u'.wmv']}
-        self.audio_extensions_list = []
-        self.video_extensions_list = []
-        mimetypes.init()
-        for mimetype in Phonon.BackendCapabilities.availableMimeTypes():
-            mimetype = unicode(mimetype)
-            if mimetype.startswith(u'audio/'):
-                self._addToList(self.audio_extensions_list, mimetype)
-            elif mimetype.startswith(u'video/'):
-                self._addToList(self.video_extensions_list, mimetype)
+        self.audio_extensions_list = \
+            self.mediaController.get_audio_extensions_list()
+        for ext in self.audio_extensions_list:
+            self.serviceManager.supportedSuffixes(ext[2:])
+        self.video_extensions_list = \
+            self.mediaController.get_video_extensions_list()
+        for ext in self.video_extensions_list:
+            self.serviceManager.supportedSuffixes(ext[2:])
 
-    def _addToList(self, list, mimetype):
-        # Add all extensions which mimetypes provides us for supported types.
-        extensions = mimetypes.guess_all_extensions(unicode(mimetype))
-        for extension in extensions:
-            ext = u'*%s' % extension
-            if ext not in list:
-                list.append(ext)
-                self.serviceManager.supportedSuffixes(extension[1:])
-        log.info(u'MediaPlugin: %s extensions: %s' % (mimetype,
-            u' '.join(extensions)))
-        # Add extensions for this mimetype from self.additional_extensions.
-        # This hack clears mimetypes' and operating system's shortcomings
-        # by providing possibly missing extensions.
-        if mimetype in self.additional_extensions.keys():
-            for extension in self.additional_extensions[mimetype]:
-                ext = u'*%s' % extension
-                if ext not in list:
-                    list.append(ext)
-                    self.serviceManager.supportedSuffixes(extension[1:])
-            log.info(u'MediaPlugin: %s additional extensions: %s' % (mimetype,
-                u' '.join(self.additional_extensions[mimetype])))
+    def getSettingsTab(self, parent):
+        """
+        Create the settings Tab
+        """
+        visible_name = self.getString(StringContent.VisibleName)
+        return MediaTab(parent, self.name, visible_name[u'title'],
+            self.mediaController.mediaPlayers, self.icon_path)
 
     def about(self):
         about_text = translate('MediaPlugin', '<strong>Media Plugin</strong>'
@@ -123,3 +92,29 @@
                 'Add the selected media to the service.')
         }
         self.setPluginUiTextStrings(tooltips)
+
+    def finalise(self):
+        """
+        Time to tidy up on exit
+        """
+        log.info(u'Media Finalising')
+        self.mediaController.finalise()
+        Plugin.finalise(self)
+
+    def getDisplayCss(self):
+        """
+        Add css style sheets to htmlbuilder
+        """
+        return self.mediaController.get_media_display_css()
+
+    def getDisplayJavaScript(self):
+        """
+        Add javascript functions to htmlbuilder
+        """
+        return self.mediaController.get_media_display_javascript()
+
+    def getDisplayHtml(self):
+        """
+        Add html code to htmlbuilder
+        """
+        return self.mediaController.get_media_display_html()

=== removed file 'resources/forms/mediafilesdialog.ui'
--- resources/forms/mediafilesdialog.ui	2011-08-23 21:48:46 +0000
+++ resources/forms/mediafilesdialog.ui	1970-01-01 00:00:00 +0000
@@ -1,95 +0,0 @@
-<?xml version="1.0" encoding="UTF-8"?>
-<ui version="4.0">
- <class>MediaFilesDialog</class>
- <widget class="QDialog" name="MediaFilesDialog">
-  <property name="windowModality">
-   <enum>Qt::ApplicationModal</enum>
-  </property>
-  <property name="geometry">
-   <rect>
-    <x>0</x>
-    <y>0</y>
-    <width>400</width>
-    <height>300</height>
-   </rect>
-  </property>
-  <property name="windowTitle">
-   <string>Select Media File(s)</string>
-  </property>
-  <property name="modal">
-   <bool>true</bool>
-  </property>
-  <layout class="QVBoxLayout" name="filesVerticalLayout">
-   <property name="spacing">
-    <number>8</number>
-   </property>
-   <property name="margin">
-    <number>8</number>
-   </property>
-   <item>
-    <widget class="QLabel" name="selectLabel">
-     <property name="text">
-      <string>Select one or more audio files from the list below, and click OK to import them into this song.</string>
-     </property>
-     <property name="wordWrap">
-      <bool>true</bool>
-     </property>
-    </widget>
-   </item>
-   <item>
-    <widget class="QListView" name="fileListView">
-     <property name="alternatingRowColors">
-      <bool>true</bool>
-     </property>
-    </widget>
-   </item>
-   <item>
-    <widget class="QDialogButtonBox" name="buttonBox">
-     <property name="orientation">
-      <enum>Qt::Horizontal</enum>
-     </property>
-     <property name="standardButtons">
-      <set>QDialogButtonBox::Cancel|QDialogButtonBox::Ok</set>
-     </property>
-    </widget>
-   </item>
-  </layout>
- </widget>
- <resources>
-  <include location="../images/openlp-2.qrc"/>
- </resources>
- <connections>
-  <connection>
-   <sender>buttonBox</sender>
-   <signal>accepted()</signal>
-   <receiver>MediaFilesDialog</receiver>
-   <slot>accept()</slot>
-   <hints>
-    <hint type="sourcelabel">
-     <x>248</x>
-     <y>254</y>
-    </hint>
-    <hint type="destinationlabel">
-     <x>157</x>
-     <y>274</y>
-    </hint>
-   </hints>
-  </connection>
-  <connection>
-   <sender>buttonBox</sender>
-   <signal>rejected()</signal>
-   <receiver>MediaFilesDialog</receiver>
-   <slot>reject()</slot>
-   <hints>
-    <hint type="sourcelabel">
-     <x>316</x>
-     <y>260</y>
-    </hint>
-    <hint type="destinationlabel">
-     <x>286</x>
-     <y>274</y>
-    </hint>
-   </hints>
-  </connection>
- </connections>
-</ui>

=== added file 'resources/pyinstaller/hook-openlp.core.ui.media.py'
--- resources/pyinstaller/hook-openlp.core.ui.media.py	1970-01-01 00:00:00 +0000
+++ resources/pyinstaller/hook-openlp.core.ui.media.py	2011-11-30 20:32:27 +0000
@@ -0,0 +1,30 @@
+# -*- coding: utf-8 -*-
+# vim: autoindent shiftwidth=4 expandtab textwidth=80 tabstop=4 softtabstop=4
+
+###############################################################################
+# OpenLP - Open Source Lyrics Projection                                      #
+# --------------------------------------------------------------------------- #
+# Copyright (c) 2008-2011 Raoul Snyman                                        #
+# Portions copyright (c) 2008-2011 Tim Bentley, Jonathan Corwin, Michael      #
+# Gorven, Scott Guerrieri, Matthias Hub, Meinert Jordan, Armin Köhler,        #
+# Andreas Preikschat, Mattias Põldaru, Christian Richter, Philip Ridout,      #
+# Jeffrey Smith, Maikel Stuivenberg, Martin Thompson, Jon Tibble, Frode       #
+# Woldsund                                                                    #
+# --------------------------------------------------------------------------- #
+# This program is free software; you can redistribute it and/or modify it     #
+# under the terms of the GNU General Public License as published by the Free  #
+# Software Foundation; version 2 of the License.                              #
+#                                                                             #
+# This program is distributed in the hope that it will be useful, but WITHOUT #
+# ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or       #
+# FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for    #
+# more details.                                                               #
+#                                                                             #
+# You should have received a copy of the GNU General Public License along     #
+# with this program; if not, write to the Free Software Foundation, Inc., 59  #
+# Temple Place, Suite 330, Boston, MA 02111-1307 USA                          #
+###############################################################################
+
+hiddenimports = ['openlp.core.ui.media.phononplayer',
+                 'openlp.core.ui.media.vlcplayer',
+                 'openlp.core.ui.media.webkitplayer']

=== modified file 'scripts/windows-builder.py'
--- scripts/windows-builder.py	2011-10-28 13:28:57 +0000
+++ scripts/windows-builder.py	2011-11-30 20:32:27 +0000
@@ -233,6 +233,19 @@
                 copy(os.path.join(root, filename),
                     os.path.join(dest_path, filename))
 
+def copy_media_player():
+    print u'Copying media player...'
+    source = os.path.join(source_path, u'core', u'ui', u'media')
+    dest = os.path.join(dist_path, u'core', u'ui', u'media')
+    for root, dirs, files in os.walk(source):
+        for filename in files:
+            if not filename.endswith(u'.pyc'):
+                dest_path = os.path.join(dest, root[len(source)+1:])
+                if not os.path.exists(dest_path):
+                    os.makedirs(dest_path)
+                copy(os.path.join(root, filename),
+                    os.path.join(dest_path, filename))
+
 def copy_windows_files():
     print u'Copying extra files for Windows...'
     copy(os.path.join(winres_path, u'OpenLP.ico'),
@@ -355,6 +368,7 @@
     run_pyinstaller()
     write_version_file()
     copy_plugins()
+    copy_media_player()
     if os.path.exists(manual_path):
         run_sphinx()
         run_htmlhelp()


Follow ups