Added support for audio cds and udisks2

This commit is contained in:
Tomas Groth 2014-05-23 15:39:25 +02:00
parent d9c6e2a228
commit 06825e4aeb
4 changed files with 160 additions and 68 deletions

View File

@ -464,7 +464,10 @@ class MediaController(RegistryMixin, OpenLPMixin, RegistryProperties):
# Setup media info # Setup media info
controller.media_info = MediaInfo() controller.media_info = MediaInfo()
controller.media_info.file_info = QtCore.QFileInfo(filename) controller.media_info.file_info = QtCore.QFileInfo(filename)
controller.media_info.media_type = MediaType.DVD if audio_track == -1 and subtitle_track == -1:
controller.media_info.media_type = MediaType.CD
else:
controller.media_info.media_type = MediaType.DVD
controller.media_info.start_time = start/1000 controller.media_info.start_time = start/1000
controller.media_info.end_time = end/1000 controller.media_info.end_time = end/1000
controller.media_info.length = (end - start)/1000 controller.media_info.length = (end - start)/1000
@ -489,7 +492,10 @@ class MediaController(RegistryMixin, OpenLPMixin, RegistryProperties):
vlc_player.load(display) vlc_player.load(display)
self.resize(display, vlc_player) self.resize(display, vlc_player)
self.current_media_players[controller.controller_type] = vlc_player self.current_media_players[controller.controller_type] = vlc_player
controller.media_info.media_type = MediaType.DVD if audio_track == -1 and subtitle_track == -1:
controller.media_info.media_type = MediaType.CD
else:
controller.media_info.media_type = MediaType.DVD
return True return True
def _check_file_type(self, controller, display, service_item): def _check_file_type(self, controller, display, service_item):

View File

@ -166,7 +166,19 @@ class VlcPlayer(MediaPlayer):
file_path = str(controller.media_info.file_info.absoluteFilePath()) file_path = str(controller.media_info.file_info.absoluteFilePath())
path = os.path.normcase(file_path) path = os.path.normcase(file_path)
# create the media # create the media
display.vlc_media = display.vlc_instance.media_new_path(path) if controller.media_info.media_type == MediaType.CD:
display.vlc_media = display.vlc_instance.media_new_location('cdda://' + path)
display.vlc_media_player.set_media(display.vlc_media)
display.vlc_media_player.play()
# Wait for media to start playing. In this case VLC actually returns an error.
self.media_state_wait(display, vlc.State.Playing)
# If subitems exists, this is a CD
audio_cd_tracks = display.vlc_media.subitems()
if not audio_cd_tracks or audio_cd_tracks.count() < 1:
return False
display.vlc_media = audio_cd_tracks.item_at_index(controller.media_info.title_track)
else:
display.vlc_media = display.vlc_instance.media_new_path(path)
# put the media in the media player # put the media in the media player
display.vlc_media_player.set_media(display.vlc_media) display.vlc_media_player.set_media(display.vlc_media)
# parse the metadata of the file # parse the metadata of the file

View File

@ -150,6 +150,40 @@ class MediaClipSelectorForm(QtGui.QDialog, Ui_MediaClipSelector):
self.timer.timeout.connect(self.update_position) self.timer.timeout.connect(self.update_position)
self.timer.start(100) self.timer.start(100)
self.find_optical_devices() self.find_optical_devices()
self.audio_cd = False
self.audio_cd_tracks = None
def detect_audio_cd(self, path):
"""
Detects is the given path is an audio CD
:param path: Path to the device to be tested.
:return: True if it was an audio CD else False.
"""
# Detect by trying to play it as a CD
self.vlc_media = self.vlc_instance.media_new_location('cdda://' + path)
self.vlc_media_player.set_media(self.vlc_media)
self.vlc_media_player.play()
# Wait for media to start playing. In this case VLC actually returns an error.
self.media_state_wait(vlc.State.Playing)
self.vlc_media_player.pause()
# If subitems exists, this is a CD
self.audio_cd_tracks = self.vlc_media.subitems()
if not self.audio_cd_tracks or self.audio_cd_tracks.count() < 1:
return False
# Insert into title_combo_box
self.title_combo_box.clear()
for i in range(self.audio_cd_tracks.count()):
item = self.audio_cd_tracks.item_at_index(i)
item_title = item.get_meta(vlc.Meta.Title)
self.title_combo_box.addItem(item_title, i)
self.vlc_media_player.set_media(self.audio_cd_tracks.item_at_index(0))
self.audio_cd = True
self.title_combo_box.setDisabled(False)
self.title_combo_box.setCurrentIndex(0)
self.on_title_combo_box_currentIndexChanged(0)
return True
@QtCore.pyqtSlot(bool) @QtCore.pyqtSlot(bool)
def on_load_disc_pushbutton_clicked(self, clicked): def on_load_disc_pushbutton_clicked(self, clicked):
@ -172,7 +206,7 @@ class MediaClipSelectorForm(QtGui.QDialog, Ui_MediaClipSelector):
'Given path does not exists')) 'Given path does not exists'))
self.toggle_disable_load_media(False) self.toggle_disable_load_media(False)
return return
self.vlc_media = self.vlc_instance.media_new_path(path) self.vlc_media = self.vlc_instance.media_new_location(path)
if not self.vlc_media: if not self.vlc_media:
log.debug('vlc media player is none') log.debug('vlc media player is none')
critical_error_message_box(message=translate('MediaPlugin.MediaClipSelectorForm', critical_error_message_box(message=translate('MediaPlugin.MediaClipSelectorForm',
@ -191,25 +225,28 @@ class MediaClipSelectorForm(QtGui.QDialog, Ui_MediaClipSelector):
return return
self.vlc_media_player.audio_set_mute(True) self.vlc_media_player.audio_set_mute(True)
if not self.media_state_wait(vlc.State.Playing): if not self.media_state_wait(vlc.State.Playing):
critical_error_message_box(message=translate('MediaPlugin.MediaClipSelectorForm', # Tests if this is an audio CD
'VLC player failed playing the media')) if not self.detect_audio_cd(path):
self.toggle_disable_load_media(False) critical_error_message_box(message=translate('MediaPlugin.MediaClipSelectorForm',
return 'VLC player failed playing the media'))
self.toggle_disable_load_media(False)
return
self.vlc_media_player.pause() self.vlc_media_player.pause()
self.vlc_media_player.set_time(0) self.vlc_media_player.set_time(0)
# Get titles, insert in combobox if not self.audio_cd:
titles = self.vlc_media_player.video_get_title_description() # Get titles, insert in combobox
self.title_combo_box.clear() titles = self.vlc_media_player.video_get_title_description()
for title in titles: self.title_combo_box.clear()
self.title_combo_box.addItem(title[1].decode(), title[0]) for title in titles:
# Main title is usually title #1 self.title_combo_box.addItem(title[1].decode(), title[0])
if len(titles) > 1: # Main title is usually title #1
self.title_combo_box.setCurrentIndex(1) if len(titles) > 1:
else: self.title_combo_box.setCurrentIndex(1)
self.title_combo_box.setCurrentIndex(0) else:
# Enable audio track combobox if anything is in it self.title_combo_box.setCurrentIndex(0)
if len(titles) > 0: # Enable audio track combobox if anything is in it
self.title_combo_box.setDisabled(False) if len(titles) > 0:
self.title_combo_box.setDisabled(False)
self.toggle_disable_load_media(False) self.toggle_disable_load_media(False)
@QtCore.pyqtSlot(bool) @QtCore.pyqtSlot(bool)
@ -321,35 +358,53 @@ class MediaClipSelectorForm(QtGui.QDialog, Ui_MediaClipSelector):
log.debug('in on_title_combo_box_changed, index: %d', index) log.debug('in on_title_combo_box_changed, index: %d', index)
if not self.vlc_media_player: if not self.vlc_media_player:
return return
self.vlc_media_player.set_title(index) if self.audio_cd:
self.vlc_media_player.set_time(0) self.vlc_media = self.audio_cd_tracks.item_at_index(index)
self.vlc_media_player.play() self.vlc_media_player.set_media(self.vlc_media)
self.vlc_media_player.audio_set_mute(True) self.vlc_media_player.set_time(0)
if not self.media_state_wait(vlc.State.Playing): self.vlc_media_player.play()
return self.vlc_media_player.audio_set_mute(True)
# pause if not self.media_state_wait(vlc.State.Playing):
self.vlc_media_player.pause() return
self.vlc_media_player.set_time(0) # pause
# Get audio tracks, insert in combobox self.vlc_media_player.pause()
audio_tracks = self.vlc_media_player.audio_get_track_description() self.vlc_media_player.set_time(0)
self.audio_tracks_combobox.clear() self.vlc_media_player.audio_set_mute(False)
for audio_track in audio_tracks: self.toggle_disable_player(False)
self.audio_tracks_combobox.addItem(audio_track[1].decode(), audio_track[0]) else:
# Enable audio track combobox if anything is in it self.vlc_media_player.set_title(index)
if len(audio_tracks) > 0: self.vlc_media_player.set_time(0)
self.audio_tracks_combobox.setDisabled(False) self.vlc_media_player.play()
# First track is "deactivated", so set to next if it exists self.vlc_media_player.audio_set_mute(True)
if len(audio_tracks) > 1: if not self.media_state_wait(vlc.State.Playing):
self.audio_tracks_combobox.setCurrentIndex(1) return
# Get subtitle tracks, insert in combobox # pause
subtitles_tracks = self.vlc_media_player.video_get_spu_description() self.vlc_media_player.pause()
self.subtitle_tracks_combobox.clear() self.vlc_media_player.set_time(0)
for subtitle_track in subtitles_tracks: # Get audio tracks, insert in combobox
self.subtitle_tracks_combobox.addItem(subtitle_track[1].decode(), subtitle_track[0]) audio_tracks = self.vlc_media_player.audio_get_track_description()
# Enable subtitle track combobox is anything in it self.audio_tracks_combobox.clear()
if len(subtitles_tracks) > 0: for audio_track in audio_tracks:
self.subtitle_tracks_combobox.setDisabled(False) self.audio_tracks_combobox.addItem(audio_track[1].decode(), audio_track[0])
self.vlc_media_player.audio_set_mute(False) # Enable audio track combobox if anything is in it
if len(audio_tracks) > 0:
self.audio_tracks_combobox.setDisabled(False)
# First track is "deactivated", so set to next if it exists
if len(audio_tracks) > 1:
self.audio_tracks_combobox.setCurrentIndex(1)
# Get subtitle tracks, insert in combobox
subtitles_tracks = self.vlc_media_player.video_get_spu_description()
self.subtitle_tracks_combobox.clear()
for subtitle_track in subtitles_tracks:
self.subtitle_tracks_combobox.addItem(subtitle_track[1].decode(), subtitle_track[0])
# Enable subtitle track combobox is anything in it
if len(subtitles_tracks) > 0:
self.subtitle_tracks_combobox.setDisabled(False)
self.vlc_media_player.audio_set_mute(False)
# If a title or audio track is available the player is enabled
if self.title_combo_box.count() > 0 or len(audio_tracks) > 0:
self.toggle_disable_player(False)
# Set media length info
self.playback_length = self.vlc_media_player.get_length() self.playback_length = self.vlc_media_player.get_length()
self.position_horizontalslider.setMaximum(self.playback_length) self.position_horizontalslider.setMaximum(self.playback_length)
# setup start and end time # setup start and end time
@ -359,9 +414,6 @@ class MediaClipSelectorForm(QtGui.QDialog, Ui_MediaClipSelector):
self.start_timeedit.setMaximumTime(playback_length_time) self.start_timeedit.setMaximumTime(playback_length_time)
self.end_timeedit.setMaximumTime(playback_length_time) self.end_timeedit.setMaximumTime(playback_length_time)
self.end_timeedit.setTime(playback_length_time) self.end_timeedit.setTime(playback_length_time)
# If a title or audio track is available the player is enabled
if self.title_combo_box.count() > 0 or len(audio_tracks) > 0:
self.toggle_disable_player(False)
@QtCore.pyqtSlot(int) @QtCore.pyqtSlot(int)
def on_audio_tracks_combobox_currentIndexChanged(self, index): def on_audio_tracks_combobox_currentIndexChanged(self, index):
@ -465,11 +517,15 @@ class MediaClipSelectorForm(QtGui.QDialog, Ui_MediaClipSelector):
end_time.second() * 1000 + \ end_time.second() * 1000 + \
end_time.msec() end_time.msec()
title = self.title_combo_box.itemData(self.title_combo_box.currentIndex()) title = self.title_combo_box.itemData(self.title_combo_box.currentIndex())
audio_track = self.audio_tracks_combobox.itemData(self.audio_tracks_combobox.currentIndex())
subtitle_track = self.subtitle_tracks_combobox.itemData(self.subtitle_tracks_combobox.currentIndex())
path = self.media_path_combobox.currentText() path = self.media_path_combobox.currentText()
optical = 'optical:' + str(title) + ':' + str(audio_track) + ':' + str(subtitle_track) + ':' + str( optical = ''
start_time_ms) + ':' + str(end_time_ms) + ':' + path if self.audio_cd:
optical = 'optical:' + str(title) + ':-1:-1:' + str(start_time_ms) + ':' + str(end_time_ms) + ':' + path
else:
audio_track = self.audio_tracks_combobox.itemData(self.audio_tracks_combobox.currentIndex())
subtitle_track = self.subtitle_tracks_combobox.itemData(self.subtitle_tracks_combobox.currentIndex())
optical = 'optical:' + str(title) + ':' + str(audio_track) + ':' + str(subtitle_track) + ':' + str(
start_time_ms) + ':' + str(end_time_ms) + ':' + path
self.media_item.add_optical_clip(optical) self.media_item.add_optical_clip(optical)
def media_state_wait(self, media_state): def media_state_wait(self, media_state):
@ -513,16 +569,34 @@ class MediaClipSelectorForm(QtGui.QDialog, Ui_MediaClipSelector):
elif sys.platform.startswith('linux'): elif sys.platform.startswith('linux'):
# Get disc devices from dbus and find the ones that are optical # Get disc devices from dbus and find the ones that are optical
bus = dbus.SystemBus() bus = dbus.SystemBus()
udev_manager_obj = bus.get_object('org.freedesktop.UDisks', '/org/freedesktop/UDisks') try:
udev_manager = dbus.Interface(udev_manager_obj, 'org.freedesktop.UDisks') udev_manager_obj = bus.get_object('org.freedesktop.UDisks', '/org/freedesktop/UDisks')
for dev in udev_manager.EnumerateDevices(): udev_manager = dbus.Interface(udev_manager_obj, 'org.freedesktop.UDisks')
device_obj = bus.get_object("org.freedesktop.UDisks", dev) for dev in udev_manager.EnumerateDevices():
device_props = dbus.Interface(device_obj, dbus.PROPERTIES_IFACE) device_obj = bus.get_object("org.freedesktop.UDisks", dev)
if device_props.Get('org.freedesktop.UDisks.Device', 'DeviceIsDrive'): device_props = dbus.Interface(device_obj, dbus.PROPERTIES_IFACE)
drive_props = device_props.Get('org.freedesktop.UDisks.Device', 'DriveMediaCompatibility') if device_props.Get('org.freedesktop.UDisks.Device', 'DeviceIsDrive'):
if any('optical' in prop for prop in drive_props): drive_props = device_props.Get('org.freedesktop.UDisks.Device', 'DriveMediaCompatibility')
self.media_path_combobox.addItem(device_props.Get('org.freedesktop.UDisks.Device', if any('optical' in prop for prop in drive_props):
'DeviceFile')) self.media_path_combobox.addItem(device_props.Get('org.freedesktop.UDisks.Device',
'DeviceFile'))
return
except dbus.exceptions.DBusException:
log.debug('could not use udisks, will try udisks2')
udev_manager_obj = bus.get_object('org.freedesktop.UDisks2', '/org/freedesktop/UDisks2')
udev_manager = dbus.Interface(udev_manager_obj, 'org.freedesktop.DBus.ObjectManager')
for k,v in udev_manager.GetManagedObjects().items():
drive_info = v.get('org.freedesktop.UDisks2.Drive', {})
drive_props = drive_info.get('MediaCompatibility')
if drive_props and any('optical' in prop for prop in drive_props):
for device in udev_manager.GetManagedObjects().values():
if dbus.String('org.freedesktop.UDisks2.Block') in device:
if device[dbus.String('org.freedesktop.UDisks2.Block')][dbus.String('Drive')] == k:
block_file = ''
for c in device[dbus.String('org.freedesktop.UDisks2.Block')][dbus.String('PreferredDevice')]:
if chr(c) != '\x00':
block_file += chr(c)
self.media_path_combobox.addItem(block_file)
elif sys.platform.startswith('darwin'): elif sys.platform.startswith('darwin'):
# Look for DVD folders in devices to find optical devices # Look for DVD folders in devices to find optical devices
volumes = os.listdir('/Volumes') volumes = os.listdir('/Volumes')

View File

@ -428,6 +428,6 @@ class MediaMediaItem(MediaManagerItem, RegistryProperties):
:return: Time string in format: hh.mm.ss,ttt :return: Time string in format: hh.mm.ss,ttt
""" """
seconds, millis = divmod(milliseconds, 1000) seconds, millis = divmod(milliseconds, 1000)
minutes, seconds = divmod(millis, 60) minutes, seconds = divmod(seconds, 60)
hours, minutes = divmod(minutes, 60) hours, minutes = divmod(minutes, 60)
return "%02d:%02d:%02d,%03d" % (hours, minutes, seconds, millis) return "%02d:%02d:%02d,%03d" % (hours, minutes, seconds, millis)