745 lines
33 KiB
Python
745 lines
33 KiB
Python
# -*- coding: utf-8 -*-
|
|
#
|
|
# gPodder - A media aggregator and podcast client
|
|
# Copyright (c) 2005-2018 The gPodder Team
|
|
#
|
|
# gPodder 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; either version 3 of the License, or
|
|
# (at your option) any later version.
|
|
#
|
|
# gPodder 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, see <http://www.gnu.org/licenses/>.
|
|
#
|
|
|
|
import html
|
|
import logging
|
|
|
|
from gi.repository import Gdk, Gtk, Pango
|
|
|
|
import gpodder
|
|
from gpodder import util, vimeo, youtube
|
|
from gpodder.gtkui.desktopfile import PlayerListModel
|
|
from gpodder.gtkui.interface.common import (BuilderWidget, TreeViewHelper,
|
|
show_message_dialog)
|
|
from gpodder.gtkui.interface.configeditor import gPodderConfigEditor
|
|
|
|
logger = logging.getLogger(__name__)
|
|
|
|
_ = gpodder.gettext
|
|
N_ = gpodder.ngettext
|
|
|
|
|
|
class NewEpisodeActionList(Gtk.ListStore):
|
|
C_CAPTION, C_AUTO_DOWNLOAD = list(range(2))
|
|
|
|
ACTION_NONE, ACTION_ASK, ACTION_MINIMIZED, ACTION_ALWAYS = list(range(4))
|
|
|
|
def __init__(self, config):
|
|
Gtk.ListStore.__init__(self, str, str)
|
|
self._config = config
|
|
self.append((_('Do nothing'), 'ignore'))
|
|
self.append((_('Show episode list'), 'show'))
|
|
self.append((_('Add to download list'), 'queue'))
|
|
self.append((_('Download immediately'), 'download'))
|
|
|
|
def get_index(self):
|
|
for index, row in enumerate(self):
|
|
if self._config.ui.gtk.new_episodes == row[self.C_AUTO_DOWNLOAD]:
|
|
return index
|
|
|
|
return 1 # Some sane default
|
|
|
|
def set_index(self, index):
|
|
self._config.ui.gtk.new_episodes = self[index][self.C_AUTO_DOWNLOAD]
|
|
|
|
|
|
class DeviceTypeActionList(Gtk.ListStore):
|
|
C_CAPTION, C_DEVICE_TYPE = list(range(2))
|
|
|
|
def __init__(self, config):
|
|
Gtk.ListStore.__init__(self, str, str)
|
|
self._config = config
|
|
self.append((_('None'), 'none'))
|
|
self.append((_('iPod'), 'ipod'))
|
|
self.append((_('Filesystem-based'), 'filesystem'))
|
|
|
|
def get_index(self):
|
|
for index, row in enumerate(self):
|
|
if self._config.device_sync.device_type == row[self.C_DEVICE_TYPE]:
|
|
return index
|
|
return 0 # Some sane default
|
|
|
|
def set_index(self, index):
|
|
self._config.device_sync.device_type = self[index][self.C_DEVICE_TYPE]
|
|
|
|
|
|
class OnSyncActionList(Gtk.ListStore):
|
|
C_CAPTION, C_ON_SYNC_DELETE, C_ON_SYNC_MARK_PLAYED = list(range(3))
|
|
ACTION_NONE, ACTION_ASK, ACTION_MINIMIZED, ACTION_ALWAYS = list(range(4))
|
|
|
|
def __init__(self, config):
|
|
Gtk.ListStore.__init__(self, str, bool, bool)
|
|
self._config = config
|
|
self.append((_('Do nothing'), False, False))
|
|
self.append((_('Mark as played'), False, True))
|
|
self.append((_('Delete from gPodder'), True, False))
|
|
|
|
def get_index(self):
|
|
for index, row in enumerate(self):
|
|
if (self._config.device_sync.after_sync.delete_episodes
|
|
and row[self.C_ON_SYNC_DELETE]):
|
|
return index
|
|
if (self._config.device_sync.after_sync.mark_episodes_played
|
|
and row[self.C_ON_SYNC_MARK_PLAYED] and not
|
|
self._config.device_sync.after_sync.delete_episodes):
|
|
return index
|
|
return 0 # Some sane default
|
|
|
|
def set_index(self, index):
|
|
self._config.device_sync.after_sync.delete_episodes = self[index][self.C_ON_SYNC_DELETE]
|
|
self._config.device_sync.after_sync.mark_episodes_played = self[index][self.C_ON_SYNC_MARK_PLAYED]
|
|
|
|
|
|
class YouTubeVideoFormatListModel(Gtk.ListStore):
|
|
C_CAPTION, C_ID = list(range(2))
|
|
|
|
def __init__(self, config):
|
|
Gtk.ListStore.__init__(self, str, int)
|
|
self._config = config
|
|
|
|
if self._config.youtube.preferred_fmt_ids:
|
|
caption = _('Custom (%(format_ids)s)') % {
|
|
'format_ids': ', '.join(str(x) for x in self._config.youtube.preferred_fmt_ids),
|
|
}
|
|
self.append((caption, 0))
|
|
|
|
for id, (fmt_id, path, description) in youtube.formats:
|
|
self.append((description, id))
|
|
|
|
def get_index(self):
|
|
for index, row in enumerate(self):
|
|
if self._config.youtube.preferred_fmt_id == row[self.C_ID]:
|
|
return index
|
|
return 0
|
|
|
|
def set_index(self, index):
|
|
self._config.youtube.preferred_fmt_id = self[index][self.C_ID]
|
|
|
|
|
|
class YouTubeVideoHLSFormatListModel(Gtk.ListStore):
|
|
C_CAPTION, C_ID = list(range(2))
|
|
|
|
def __init__(self, config):
|
|
Gtk.ListStore.__init__(self, str, int)
|
|
self._config = config
|
|
|
|
if self._config.youtube.preferred_hls_fmt_ids:
|
|
caption = _('Custom (%(format_ids)s)') % {
|
|
'format_ids': ', '.join(str(x) for x in self._config.youtube.preferred_hls_fmt_ids),
|
|
}
|
|
self.append((caption, 0))
|
|
|
|
for id, (fmt_id, path, description) in youtube.hls_formats:
|
|
self.append((description, id))
|
|
|
|
def get_index(self):
|
|
for index, row in enumerate(self):
|
|
if self._config.youtube.preferred_hls_fmt_id == row[self.C_ID]:
|
|
return index
|
|
return 0
|
|
|
|
def set_index(self, index):
|
|
self._config.youtube.preferred_hls_fmt_id = self[index][self.C_ID]
|
|
|
|
|
|
class VimeoVideoFormatListModel(Gtk.ListStore):
|
|
C_CAPTION, C_ID = list(range(2))
|
|
|
|
def __init__(self, config):
|
|
Gtk.ListStore.__init__(self, str, str)
|
|
self._config = config
|
|
|
|
for fileformat, description in vimeo.FORMATS:
|
|
self.append((description, fileformat))
|
|
|
|
def get_index(self):
|
|
for index, row in enumerate(self):
|
|
if self._config.vimeo.fileformat == row[self.C_ID]:
|
|
return index
|
|
return 0
|
|
|
|
def set_index(self, index):
|
|
value = self[index][self.C_ID]
|
|
if value is not None:
|
|
self._config.vimeo.fileformat = value
|
|
|
|
|
|
class gPodderPreferences(BuilderWidget):
|
|
C_TOGGLE, C_LABEL, C_EXTENSION, C_SHOW_TOGGLE = list(range(4))
|
|
|
|
def new(self):
|
|
self.gPodderPreferences.set_transient_for(self.parent_widget)
|
|
for cb in (self.combo_audio_player_app, self.combo_video_player_app):
|
|
cellrenderer = Gtk.CellRendererPixbuf()
|
|
cb.pack_start(cellrenderer, False)
|
|
cb.add_attribute(cellrenderer, 'pixbuf', PlayerListModel.C_ICON)
|
|
cellrenderer = Gtk.CellRendererText()
|
|
cellrenderer.set_property('ellipsize', Pango.EllipsizeMode.END)
|
|
cb.pack_start(cellrenderer, True)
|
|
cb.add_attribute(cellrenderer, 'markup', PlayerListModel.C_NAME)
|
|
cb.set_row_separator_func(PlayerListModel.is_separator)
|
|
|
|
self.audio_player_model = self.user_apps_reader.get_model('audio')
|
|
self.combo_audio_player_app.set_model(self.audio_player_model)
|
|
index = self.audio_player_model.get_index(self._config.player.audio)
|
|
self.combo_audio_player_app.set_active(index)
|
|
|
|
self.video_player_model = self.user_apps_reader.get_model('video')
|
|
self.combo_video_player_app.set_model(self.video_player_model)
|
|
index = self.video_player_model.get_index(self._config.player.video)
|
|
self.combo_video_player_app.set_active(index)
|
|
|
|
self.preferred_youtube_format_model = YouTubeVideoFormatListModel(self._config)
|
|
self.combobox_preferred_youtube_format.set_model(self.preferred_youtube_format_model)
|
|
cellrenderer = Gtk.CellRendererText()
|
|
cellrenderer.set_property('ellipsize', Pango.EllipsizeMode.END)
|
|
self.combobox_preferred_youtube_format.pack_start(cellrenderer, True)
|
|
self.combobox_preferred_youtube_format.add_attribute(cellrenderer, 'text', self.preferred_youtube_format_model.C_CAPTION)
|
|
self.combobox_preferred_youtube_format.set_active(self.preferred_youtube_format_model.get_index())
|
|
|
|
self.preferred_youtube_hls_format_model = YouTubeVideoHLSFormatListModel(self._config)
|
|
self.combobox_preferred_youtube_hls_format.set_model(self.preferred_youtube_hls_format_model)
|
|
cellrenderer = Gtk.CellRendererText()
|
|
cellrenderer.set_property('ellipsize', Pango.EllipsizeMode.END)
|
|
self.combobox_preferred_youtube_hls_format.pack_start(cellrenderer, True)
|
|
self.combobox_preferred_youtube_hls_format.add_attribute(cellrenderer, 'text', self.preferred_youtube_hls_format_model.C_CAPTION)
|
|
self.combobox_preferred_youtube_hls_format.set_active(self.preferred_youtube_hls_format_model.get_index())
|
|
|
|
self.preferred_vimeo_format_model = VimeoVideoFormatListModel(self._config)
|
|
self.combobox_preferred_vimeo_format.set_model(self.preferred_vimeo_format_model)
|
|
cellrenderer = Gtk.CellRendererText()
|
|
cellrenderer.set_property('ellipsize', Pango.EllipsizeMode.END)
|
|
self.combobox_preferred_vimeo_format.pack_start(cellrenderer, True)
|
|
self.combobox_preferred_vimeo_format.add_attribute(cellrenderer, 'text', self.preferred_vimeo_format_model.C_CAPTION)
|
|
self.combobox_preferred_vimeo_format.set_active(self.preferred_vimeo_format_model.get_index())
|
|
|
|
self._config.connect_gtk_togglebutton('ui.gtk.find_as_you_type',
|
|
self.checkbutton_find_as_you_type)
|
|
|
|
self.update_interval_presets = [0, 10, 30, 60, 2 * 60, 6 * 60, 12 * 60]
|
|
adjustment_update_interval = self.hscale_update_interval.get_adjustment()
|
|
adjustment_update_interval.set_upper(len(self.update_interval_presets) - 1)
|
|
if self._config.auto.update.frequency in self.update_interval_presets:
|
|
index = self.update_interval_presets.index(self._config.auto.update.frequency)
|
|
self.hscale_update_interval.set_value(index)
|
|
else:
|
|
# Patch in the current "custom" value into the mix
|
|
self.update_interval_presets.append(self._config.auto.update.frequency)
|
|
self.update_interval_presets.sort()
|
|
|
|
adjustment_update_interval.set_upper(len(self.update_interval_presets) - 1)
|
|
index = self.update_interval_presets.index(self._config.auto.update.frequency)
|
|
self.hscale_update_interval.set_value(index)
|
|
|
|
self._config.connect_gtk_spinbutton('limit.episodes', self.spinbutton_episode_limit)
|
|
|
|
self._config.connect_gtk_togglebutton('ui.gtk.only_added_are_new',
|
|
self.checkbutton_only_added_are_new)
|
|
|
|
self.auto_download_model = NewEpisodeActionList(self._config)
|
|
self.combo_auto_download.set_model(self.auto_download_model)
|
|
cellrenderer = Gtk.CellRendererText()
|
|
self.combo_auto_download.pack_start(cellrenderer, True)
|
|
self.combo_auto_download.add_attribute(cellrenderer, 'text', NewEpisodeActionList.C_CAPTION)
|
|
self.combo_auto_download.set_active(self.auto_download_model.get_index())
|
|
|
|
self._config.connect_gtk_togglebutton('check_connection',
|
|
self.checkbutton_check_connection)
|
|
|
|
if self._config.auto.cleanup.played:
|
|
adjustment_expiration = self.hscale_expiration.get_adjustment()
|
|
if self._config.auto.cleanup.days > adjustment_expiration.get_upper():
|
|
# Patch the adjustment to include the higher current value
|
|
adjustment_expiration.set_upper(self._config.auto.cleanup.days)
|
|
|
|
self.hscale_expiration.set_value(self._config.auto.cleanup.days)
|
|
else:
|
|
self.hscale_expiration.set_value(0)
|
|
|
|
self._config.connect_gtk_togglebutton('auto.cleanup.unplayed',
|
|
self.checkbutton_expiration_unplayed)
|
|
self._config.connect_gtk_togglebutton('auto.cleanup.unfinished',
|
|
self.checkbutton_expiration_unfinished)
|
|
|
|
self.device_type_model = DeviceTypeActionList(self._config)
|
|
self.combobox_device_type.set_model(self.device_type_model)
|
|
cellrenderer = Gtk.CellRendererText()
|
|
self.combobox_device_type.pack_start(cellrenderer, True)
|
|
self.combobox_device_type.add_attribute(cellrenderer, 'text',
|
|
DeviceTypeActionList.C_CAPTION)
|
|
self.combobox_device_type.set_active(self.device_type_model.get_index())
|
|
|
|
self.on_sync_model = OnSyncActionList(self._config)
|
|
self.combobox_on_sync.set_model(self.on_sync_model)
|
|
cellrenderer = Gtk.CellRendererText()
|
|
self.combobox_on_sync.pack_start(cellrenderer, True)
|
|
self.combobox_on_sync.add_attribute(cellrenderer, 'text', OnSyncActionList.C_CAPTION)
|
|
self.combobox_on_sync.set_active(self.on_sync_model.get_index())
|
|
|
|
self._config.connect_gtk_togglebutton('device_sync.skip_played_episodes',
|
|
self.checkbutton_skip_played_episodes)
|
|
self._config.connect_gtk_togglebutton('device_sync.playlists.create',
|
|
self.checkbutton_create_playlists)
|
|
self._config.connect_gtk_togglebutton('device_sync.playlists.two_way_sync',
|
|
self.checkbutton_delete_using_playlists)
|
|
self._config.connect_gtk_togglebutton('device_sync.delete_deleted_episodes',
|
|
self.checkbutton_delete_deleted_episodes)
|
|
|
|
# Have to do this before calling set_active on checkbutton_enable
|
|
self._enable_mygpo = self._config.mygpo.enabled
|
|
|
|
# Initialize the UI state with configuration settings
|
|
self.checkbutton_enable.set_active(self._config.mygpo.enabled)
|
|
self.entry_server.set_text(self._config.mygpo.server)
|
|
self.entry_username.set_text(self._config.mygpo.username)
|
|
self.entry_password.set_text(self._config.mygpo.password)
|
|
self.entry_caption.set_text(self._config.mygpo.device.caption)
|
|
|
|
# Disable mygpo sync while the dialog is open
|
|
self._config.mygpo.enabled = False
|
|
|
|
# Configure the extensions manager GUI
|
|
self.set_extension_preferences()
|
|
|
|
self._config.connect_gtk_window(self.main_window, 'preferences', True)
|
|
|
|
gpodder.user_extensions.on_ui_object_available('preferences-gtk', self)
|
|
|
|
self.inject_extensions_preferences(init=True)
|
|
|
|
self.prefs_stack.foreach(self._wrap_checkbox_labels)
|
|
|
|
def _wrap_checkbox_labels(self, w, *args):
|
|
if w.get_name().startswith("no_label_wrap"):
|
|
return
|
|
elif isinstance(w, Gtk.CheckButton):
|
|
label = w.get_child()
|
|
label.set_line_wrap(True)
|
|
elif isinstance(w, Gtk.Container):
|
|
w.foreach(self._wrap_checkbox_labels)
|
|
|
|
def inject_extensions_preferences(self, init=False):
|
|
if not init:
|
|
# remove preferences buttons for all extensions
|
|
for child in self.prefs_stack.get_children():
|
|
if child.get_name().startswith("extension."):
|
|
self.prefs_stack.remove(child)
|
|
|
|
# add preferences buttons for all extensions
|
|
result = gpodder.user_extensions.on_preferences()
|
|
if result:
|
|
for label, callback in result:
|
|
page = callback()
|
|
name = "extension." + label
|
|
page.set_name(name)
|
|
page.foreach(self._wrap_checkbox_labels)
|
|
self.prefs_stack.add_titled(page, name, label)
|
|
|
|
def _extensions_select_function(self, selection, model, path, path_currently_selected):
|
|
return model.get_value(model.get_iter(path), self.C_SHOW_TOGGLE)
|
|
|
|
def set_extension_preferences(self):
|
|
def search_equal_func(model, column, key, it):
|
|
label = model.get_value(it, self.C_LABEL)
|
|
if key.lower() in label.lower():
|
|
# from http://www.pyGtk.org/docs/pygtk/class-gtktreeview.html:
|
|
# "func should return False to indicate that the row matches
|
|
# the search criteria."
|
|
return False
|
|
|
|
return True
|
|
self.treeviewExtensions.set_search_equal_func(search_equal_func)
|
|
|
|
selection = self.treeviewExtensions.get_selection()
|
|
selection.set_select_function(self._extensions_select_function)
|
|
|
|
toggle_cell = Gtk.CellRendererToggle()
|
|
toggle_cell.connect('toggled', self.on_extensions_cell_toggled)
|
|
toggle_column = Gtk.TreeViewColumn('')
|
|
toggle_column.pack_start(toggle_cell, True)
|
|
toggle_column.add_attribute(toggle_cell, 'active', self.C_TOGGLE)
|
|
toggle_column.add_attribute(toggle_cell, 'visible', self.C_SHOW_TOGGLE)
|
|
toggle_column.set_property('min-width', 32)
|
|
self.treeviewExtensions.append_column(toggle_column)
|
|
|
|
name_cell = Gtk.CellRendererText()
|
|
name_cell.set_property('ellipsize', Pango.EllipsizeMode.END)
|
|
extension_column = Gtk.TreeViewColumn(_('Name'))
|
|
extension_column.pack_start(name_cell, True)
|
|
extension_column.add_attribute(name_cell, 'markup', self.C_LABEL)
|
|
extension_column.set_expand(True)
|
|
self.treeviewExtensions.append_column(extension_column)
|
|
|
|
self.extensions_model = Gtk.ListStore(bool, str, object, bool)
|
|
|
|
def key_func(pair):
|
|
category, container = pair
|
|
return (category, container.metadata.title)
|
|
|
|
def convert(extensions):
|
|
for container in extensions:
|
|
yield (container.metadata.category, container)
|
|
|
|
old_category = None
|
|
for category, container in sorted(convert(
|
|
gpodder.user_extensions.get_extensions()), key=key_func):
|
|
if old_category != category:
|
|
label = '<span weight="bold">%s</span>' % html.escape(category)
|
|
self.extensions_model.append((None, label, None, False))
|
|
old_category = category
|
|
|
|
label = '%s\n<small>%s</small>' % (
|
|
html.escape(container.metadata.title),
|
|
html.escape(container.metadata.description))
|
|
self.extensions_model.append((container.enabled, label, container, True))
|
|
|
|
self.treeviewExtensions.set_model(self.extensions_model)
|
|
self.treeviewExtensions.columns_autosize()
|
|
|
|
def on_treeview_extension_button_released(self, treeview, event):
|
|
if event.window != treeview.get_bin_window():
|
|
return False
|
|
|
|
if event.type == Gdk.EventType.BUTTON_RELEASE and event.button == 3:
|
|
return self.on_treeview_extension_show_context_menu(treeview, event)
|
|
|
|
return False
|
|
|
|
def on_treeview_extension_show_context_menu(self, treeview, event=None):
|
|
selection = treeview.get_selection()
|
|
model, paths = selection.get_selected_rows()
|
|
container = model.get_value(model.get_iter(paths[0]), self.C_EXTENSION)
|
|
|
|
if not container:
|
|
return
|
|
|
|
menu = Gtk.Menu()
|
|
|
|
if container.metadata.doc:
|
|
menu_item = Gtk.MenuItem(_('Documentation'))
|
|
menu_item.connect('activate', self.open_weblink,
|
|
container.metadata.doc)
|
|
menu.append(menu_item)
|
|
|
|
menu_item = Gtk.MenuItem(_('Extension info'))
|
|
menu_item.connect('activate', self.show_extension_info, model, container)
|
|
menu.append(menu_item)
|
|
|
|
if container.metadata.payment:
|
|
menu_item = Gtk.MenuItem(_('Support the author'))
|
|
menu_item.connect('activate', self.open_weblink, container.metadata.payment)
|
|
menu.append(menu_item)
|
|
|
|
menu.show_all()
|
|
if event is None:
|
|
func = TreeViewHelper.make_popup_position_func(treeview)
|
|
menu.popup(None, None, func, None, 3, Gtk.get_current_event_time())
|
|
else:
|
|
menu.popup(None, None, None, None, 3, Gtk.get_current_event_time())
|
|
|
|
return True
|
|
|
|
def on_extensions_cell_toggled(self, cell, path):
|
|
model = self.treeviewExtensions.get_model()
|
|
it = model.get_iter(path)
|
|
container = model.get_value(it, self.C_EXTENSION)
|
|
|
|
enabled_extensions = list(self._config.extensions.enabled)
|
|
new_enabled = not model.get_value(it, self.C_TOGGLE)
|
|
|
|
if new_enabled and container.name not in enabled_extensions:
|
|
enabled_extensions.append(container.name)
|
|
elif not new_enabled and container.name in enabled_extensions:
|
|
enabled_extensions.remove(container.name)
|
|
|
|
self._config.extensions.enabled = enabled_extensions
|
|
|
|
now_enabled = (container.name in self._config.extensions.enabled)
|
|
|
|
if new_enabled == now_enabled:
|
|
model.set_value(it, self.C_TOGGLE, new_enabled)
|
|
if now_enabled:
|
|
self.on_extension_enabled(container.module)
|
|
else:
|
|
self.on_extension_disabled(container.module)
|
|
self.inject_extensions_preferences()
|
|
elif container.error is not None:
|
|
if hasattr(container.error, 'message'):
|
|
error_msg = container.error.message
|
|
else:
|
|
error_msg = str(container.error)
|
|
self.show_message(error_msg,
|
|
_('Extension cannot be activated'), important=True)
|
|
model.set_value(it, self.C_TOGGLE, False)
|
|
|
|
def show_extension_info(self, w, model, container):
|
|
if not container or not model:
|
|
return
|
|
|
|
info = '\n'.join('<b>{}:</b> {}'.format(html.escape(key), html.escape(value))
|
|
for key, value in container.metadata.get_sorted()
|
|
if key not in ('title', 'description'))
|
|
|
|
self.show_message_details(container.metadata.title, container.metadata.description, info)
|
|
|
|
def open_weblink(self, w, url):
|
|
util.open_website(url)
|
|
|
|
def on_dialog_destroy(self, widget):
|
|
# Re-enable mygpo sync if the user has selected it
|
|
self._config.mygpo.enabled = self._enable_mygpo
|
|
# Make sure the device is successfully created/updated
|
|
self.mygpo_client.create_device()
|
|
# Flush settings for mygpo client now
|
|
self.mygpo_client.flush(now=True)
|
|
|
|
def on_button_close_clicked(self, widget):
|
|
self.main_window.destroy()
|
|
|
|
def on_button_advanced_clicked(self, widget):
|
|
self.main_window.destroy()
|
|
gPodderConfigEditor(self.parent_window, _config=self._config)
|
|
|
|
def on_combo_audio_player_app_changed(self, widget):
|
|
index = self.combo_audio_player_app.get_active()
|
|
self._config.player.audio = self.audio_player_model.get_command(index)
|
|
|
|
def on_combo_video_player_app_changed(self, widget):
|
|
index = self.combo_video_player_app.get_active()
|
|
self._config.player.video = self.video_player_model.get_command(index)
|
|
|
|
def on_combobox_preferred_youtube_format_changed(self, widget):
|
|
index = self.combobox_preferred_youtube_format.get_active()
|
|
self.preferred_youtube_format_model.set_index(index)
|
|
|
|
def on_combobox_preferred_youtube_hls_format_changed(self, widget):
|
|
index = self.combobox_preferred_youtube_hls_format.get_active()
|
|
self.preferred_youtube_hls_format_model.set_index(index)
|
|
|
|
def on_combobox_preferred_vimeo_format_changed(self, widget):
|
|
index = self.combobox_preferred_vimeo_format.get_active()
|
|
self.preferred_vimeo_format_model.set_index(index)
|
|
|
|
def on_button_audio_player_clicked(self, widget):
|
|
result = self.show_text_edit_dialog(_('Configure audio player'),
|
|
_('Command:'),
|
|
self._config.player.audio)
|
|
|
|
if result:
|
|
self._config.player.audio = result
|
|
index = self.audio_player_model.get_index(self._config.player.audio)
|
|
self.combo_audio_player_app.set_active(index)
|
|
|
|
def on_button_video_player_clicked(self, widget):
|
|
result = self.show_text_edit_dialog(_('Configure video player'),
|
|
_('Command:'),
|
|
self._config.player.video)
|
|
|
|
if result:
|
|
self._config.player.video = result
|
|
index = self.video_player_model.get_index(self._config.player.video)
|
|
self.combo_video_player_app.set_active(index)
|
|
|
|
def format_update_interval_value(self, scale, value):
|
|
value = int(value)
|
|
ret = None
|
|
if value == 0:
|
|
ret = _('manually')
|
|
elif value > 0 and len(self.update_interval_presets) > value:
|
|
ret = util.format_seconds_to_hour_min_sec(self.update_interval_presets[value] * 60)
|
|
else:
|
|
ret = str(value)
|
|
# bug in gtk3: value representation (pixels) must be smaller than value for highest value.
|
|
# this makes sense when formatting e.g. 0 to 1000 where '1000' is the longest
|
|
# string, but not when '10 minutes' is longer than '12 hours'
|
|
# so we replace spaces with non breaking spaces otherwise '10 minutes' is displayed as '10'
|
|
ret = ret.replace(' ', '\xa0')
|
|
return ret
|
|
|
|
def on_update_interval_value_changed(self, range):
|
|
value = int(range.get_value())
|
|
self._config.auto.update.enabled = (value > 0)
|
|
self._config.auto.update.frequency = self.update_interval_presets[value]
|
|
|
|
def on_combo_auto_download_changed(self, widget):
|
|
index = self.combo_auto_download.get_active()
|
|
self.auto_download_model.set_index(index)
|
|
|
|
def format_expiration_value(self, scale, value):
|
|
value = int(value)
|
|
if value == 0:
|
|
return _('manually')
|
|
else:
|
|
return N_('after %(count)d day', 'after %(count)d days',
|
|
value) % {'count': value}
|
|
|
|
def on_expiration_value_changed(self, range):
|
|
value = int(range.get_value())
|
|
|
|
if value == 0:
|
|
self.checkbutton_expiration_unplayed.set_active(False)
|
|
self._config.auto.cleanup.played = False
|
|
self._config.auto.cleanup.unplayed = False
|
|
else:
|
|
self._config.auto.cleanup.played = True
|
|
self._config.auto.cleanup.days = value
|
|
|
|
self.checkbutton_expiration_unplayed.set_sensitive(value > 0)
|
|
self.checkbutton_expiration_unfinished.set_sensitive(value > 0)
|
|
|
|
def on_enabled_toggled(self, widget):
|
|
# Only update indirectly (see on_dialog_destroy)
|
|
self._enable_mygpo = widget.get_active()
|
|
|
|
def on_server_changed(self, widget):
|
|
self._config.mygpo.server = widget.get_text()
|
|
|
|
def on_username_changed(self, widget):
|
|
self._config.mygpo.username = widget.get_text()
|
|
|
|
def on_password_changed(self, widget):
|
|
self._config.mygpo.password = widget.get_text()
|
|
|
|
def on_device_caption_changed(self, widget):
|
|
self._config.mygpo.device.caption = widget.get_text()
|
|
|
|
def on_button_overwrite_clicked(self, button):
|
|
title = _('Replace subscription list on server')
|
|
message = _('Remote podcasts that have not been added locally will be removed on the server. Continue?')
|
|
if self.show_confirmation(message, title):
|
|
@util.run_in_background
|
|
def thread_proc():
|
|
self._config.mygpo.enabled = True
|
|
self.on_send_full_subscriptions()
|
|
self._config.mygpo.enabled = False
|
|
|
|
def on_combobox_on_sync_changed(self, widget):
|
|
index = self.combobox_on_sync.get_active()
|
|
self.on_sync_model.set_index(index)
|
|
|
|
def on_checkbutton_create_playlists_toggled(
|
|
self, widget, device_type_changed=False):
|
|
if not widget.get_active():
|
|
self._config.device_sync.playlists.create = False
|
|
self.toggle_playlist_interface(False)
|
|
# need to read value of checkbutton from interface,
|
|
# rather than value of parameter
|
|
else:
|
|
self._config.device_sync.playlists.create = True
|
|
self.toggle_playlist_interface(True)
|
|
|
|
def toggle_playlist_interface(self, enabled):
|
|
if enabled and self._config.device_sync.device_type == 'filesystem':
|
|
self.btn_playlistfolder.set_sensitive(True)
|
|
self.btn_playlistfolder.set_label(self._config.device_sync.playlists.folder)
|
|
self.checkbutton_delete_using_playlists.set_sensitive(True)
|
|
children = self.btn_playlistfolder.get_children()
|
|
if children:
|
|
label = children.pop()
|
|
label.set_ellipsize(Pango.EllipsizeMode.START)
|
|
label.set_xalign(0.0)
|
|
else:
|
|
self.btn_playlistfolder.set_sensitive(False)
|
|
self.btn_playlistfolder.set_label('')
|
|
self.checkbutton_delete_using_playlists.set_sensitive(False)
|
|
|
|
def on_combobox_device_type_changed(self, widget):
|
|
index = self.combobox_device_type.get_active()
|
|
self.device_type_model.set_index(index)
|
|
device_type = self._config.device_sync.device_type
|
|
if device_type == 'none':
|
|
self.btn_filesystemMountpoint.set_label('')
|
|
self.btn_filesystemMountpoint.set_sensitive(False)
|
|
self.checkbutton_create_playlists.set_sensitive(False)
|
|
self.toggle_playlist_interface(False)
|
|
self.checkbutton_delete_using_playlists.set_sensitive(False)
|
|
self.combobox_on_sync.set_sensitive(False)
|
|
self.checkbutton_skip_played_episodes.set_sensitive(False)
|
|
elif device_type == 'filesystem':
|
|
self.btn_filesystemMountpoint.set_label(self._config.device_sync.device_folder or "")
|
|
self.btn_filesystemMountpoint.set_sensitive(True)
|
|
self.checkbutton_create_playlists.set_sensitive(True)
|
|
self.toggle_playlist_interface(self._config.device_sync.playlists.create)
|
|
self.combobox_on_sync.set_sensitive(True)
|
|
self.checkbutton_skip_played_episodes.set_sensitive(True)
|
|
self.checkbutton_delete_deleted_episodes.set_sensitive(True)
|
|
elif device_type == 'ipod':
|
|
self.btn_filesystemMountpoint.set_label(self._config.device_sync.device_folder)
|
|
self.btn_filesystemMountpoint.set_sensitive(True)
|
|
self.checkbutton_create_playlists.set_sensitive(False)
|
|
self.toggle_playlist_interface(False)
|
|
self.checkbutton_delete_using_playlists.set_sensitive(False)
|
|
self.combobox_on_sync.set_sensitive(False)
|
|
self.checkbutton_skip_played_episodes.set_sensitive(True)
|
|
self.checkbutton_delete_deleted_episodes.set_sensitive(True)
|
|
|
|
children = self.btn_filesystemMountpoint.get_children()
|
|
if children:
|
|
label = children.pop()
|
|
label.set_ellipsize(Pango.EllipsizeMode.START)
|
|
label.set_xalign(0.0)
|
|
|
|
def on_btn_device_mountpoint_clicked(self, widget):
|
|
fs = Gtk.FileChooserDialog(title=_('Select folder for mount point'),
|
|
action=Gtk.FileChooserAction.SELECT_FOLDER)
|
|
fs.set_local_only(False)
|
|
fs.add_button(_('_Cancel'), Gtk.ResponseType.CANCEL)
|
|
fs.add_button(_('_Open'), Gtk.ResponseType.OK)
|
|
|
|
fs.set_uri(self.btn_filesystemMountpoint.get_label() or "")
|
|
if fs.run() == Gtk.ResponseType.OK:
|
|
if self._config.device_sync.device_type == 'filesystem':
|
|
self._config.device_sync.device_folder = fs.get_uri()
|
|
elif self._config.device_sync.device_type == 'ipod':
|
|
self._config.device_sync.device_folder = fs.get_filename()
|
|
# Request an update of the mountpoint button
|
|
self.on_combobox_device_type_changed(None)
|
|
|
|
fs.destroy()
|
|
|
|
def on_btn_playlist_folder_clicked(self, widget):
|
|
fs = Gtk.FileChooserDialog(title=_('Select folder for playlists'),
|
|
action=Gtk.FileChooserAction.SELECT_FOLDER)
|
|
fs.set_local_only(False)
|
|
fs.add_button(_('_Cancel'), Gtk.ResponseType.CANCEL)
|
|
fs.add_button(_('_Open'), Gtk.ResponseType.OK)
|
|
|
|
device_folder = util.new_gio_file(self._config.device_sync.device_folder)
|
|
playlists_folder = device_folder.resolve_relative_path(self._config.device_sync.playlists.folder)
|
|
fs.set_file(playlists_folder)
|
|
|
|
while fs.run() == Gtk.ResponseType.OK:
|
|
filename = util.relpath(fs.get_uri(),
|
|
self._config.device_sync.device_folder)
|
|
if not filename:
|
|
show_message_dialog(fs, _('The playlists folder must be on the device'))
|
|
continue
|
|
|
|
if self._config.device_sync.device_type == 'filesystem':
|
|
self._config.device_sync.playlists.folder = filename
|
|
self.btn_playlistfolder.set_label(filename or "")
|
|
children = self.btn_playlistfolder.get_children()
|
|
if children:
|
|
label = children.pop()
|
|
label.set_ellipsize(Pango.EllipsizeMode.START)
|
|
label.set_xalign(0.0)
|
|
break
|
|
|
|
fs.destroy()
|