gpodder/src/gpodder/gtkui/desktop/preferences.py

592 lines
25 KiB
Python
Raw Normal View History

# -*- coding: utf-8 -*-
#
# gPodder - A media aggregator and podcast client
2012-01-09 21:19:24 +01:00
# Copyright (c) 2005-2012 Thomas Perl and 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 gtk
import pango
import cgi
2012-05-24 23:17:55 +02:00
import urlparse
import logging
logger = logging.getLogger(__name__)
import gpodder
_ = gpodder.gettext
N_ = gpodder.ngettext
from gpodder import util
from gpodder import youtube
from gpodder.gtkui.interface.common import BuilderWidget
from gpodder.gtkui.interface.configeditor import gPodderConfigEditor
from gpodder.gtkui.desktopfile import PlayerListModel
class NewEpisodeActionList(gtk.ListStore):
C_CAPTION, C_AUTO_DOWNLOAD = range(2)
ACTION_NONE, ACTION_ASK, ACTION_MINIMIZED, ACTION_ALWAYS = 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.auto_download == row[self.C_AUTO_DOWNLOAD]:
return index
return 1 # Some sane default
def set_index(self, index):
self._config.auto_download = self[index][self.C_AUTO_DOWNLOAD]
class DeviceTypeActionList(gtk.ListStore):
C_CAPTION, C_DEVICE_TYPE = range(2)
def __init__(self, config):
gtk.ListStore.__init__(self, str, str)
self._config = config
self.append((_('None'), 'none'))
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 = range(3)
ACTION_NONE, ACTION_ASK, ACTION_MINIMIZED, ACTION_ALWAYS = 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]
2012-05-24 23:17:55 +02:00
class gPodderFlattrSignIn(BuilderWidget):
def new(self):
import webkit
self.web = webkit.WebView()
self.web.connect('resource-request-starting', self.on_web_request)
self.main_window.connect('destroy', self.set_flattr_preferences)
auth_url = self.flattr.get_auth_url()
logger.info(auth_url)
self.web.open(auth_url)
self.scrolledwindow_web.add(self.web)
self.web.show()
def on_web_request(self, web_view, web_frame, web_resource, request, response):
uri = request.get_uri()
if uri.startswith(self.flattr.CALLBACK):
if not self.flattr.process_retrieved_code(uri):
2012-05-24 23:17:55 +02:00
self.show_message(query['error_description'][0], _('Error'),
important=True)
# Destroy the window later
util.idle_add(self.main_window.destroy)
def on_btn_close_clicked(self, widget):
util.idle_add(self.main_window.destroy)
class VideoFormatList(gtk.ListStore):
C_CAPTION, C_ID = 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(self.custom_format_ids),
}
self.append((caption, -1))
else:
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):
value = self[index][self.C_ID]
if value > 0:
self._config.youtube.preferred_fmt_id = value
2012-05-24 23:17:55 +02:00
class gPodderPreferences(BuilderWidget):
C_TOGGLE, C_LABEL, C_EXTENSION, C_SHOW_TOGGLE = range(4)
def new(self):
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.ELLIPSIZE_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)
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.videoplayer)
self.combo_video_player_app.set_active(index)
self.preferred_video_format_model = VideoFormatList(self._config)
self.combobox_preferred_video_format.set_model(self.preferred_video_format_model)
cellrenderer = gtk.CellRendererText()
self.combobox_preferred_video_format.pack_start(cellrenderer, True)
self.combobox_preferred_video_format.add_attribute(cellrenderer, 'text', self.preferred_video_format_model.C_CAPTION)
self.combobox_preferred_video_format.set_active(self.preferred_video_format_model.get_index())
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.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.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('max_episodes_per_feed', self.spinbutton_episode_limit)
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())
if self._config.auto_remove_played_episodes:
adjustment_expiration = self.hscale_expiration.get_adjustment()
if self._config.episode_old_age > adjustment_expiration.get_upper():
# Patch the adjustment to include the higher current value
adjustment_expiration.upper = self._config.episode_old_age
self.hscale_expiration.set_value(self._config.episode_old_age)
else:
self.hscale_expiration.set_value(0)
2013-02-01 04:36:21 +01:00
self._config.connect_gtk_togglebutton('auto_remove_unplayed_episodes',
self.checkbutton_expiration_unplayed)
self._config.connect_gtk_togglebutton('auto_remove_unfinished_episodes',
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)
2013-02-01 04:36:21 +01:00
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())
2013-02-01 04:36:21 +01:00
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)
# 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_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
2012-05-24 23:17:55 +02:00
# Initialize Flattr settings
self.set_flattr_preferences()
# Configure the extensions manager GUI
self.set_extension_preferences()
def set_extension_preferences(self):
toggle_cell = gtk.CellRendererToggle()
toggle_cell.connect('toggled', self.on_extensions_cell_toggled)
name_cell = gtk.CellRendererText()
name_cell.set_property('ellipsize', pango.ELLIPSIZE_END)
extension_column = gtk.TreeViewColumn(_('Name'))
extension_column.pack_start(toggle_cell, False)
extension_column.add_attribute(toggle_cell, 'active', self.C_TOGGLE)
extension_column.add_attribute(toggle_cell, 'visible', self.C_SHOW_TOGGLE)
extension_column.pack_start(name_cell, True)
extension_column.add_attribute(name_cell, 'markup', self.C_LABEL)
extension_column.set_clickable(False)
extension_column.set_resizable(True)
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>' % cgi.escape(category)
self.extensions_model.append((None, label, None, False))
old_category = category
label = '%s\n<small>%s</small>' % (
cgi.escape(container.metadata.title),
cgi.escape(container.metadata.description))
self.extensions_model.append((container.enabled, label, container, True))
self.treeviewExtensions.set_model(self.extensions_model)
self.treeviewExtensions.columns_autosize()
2012-05-24 23:17:55 +02:00
def set_flattr_preferences(self, widget=None):
if not self._config.flattr.token:
self.label_flattr.set_text(_('Please sign in with Flattr and Support Publishers'))
self.button_flattr_login.set_label(_('Sign in to Flattr'))
else:
flattr_user = self.flattr.get_auth_username()
self.label_flattr.set_markup(_('Logged in as <b>%(username)s</b>') % {'username': flattr_user})
self.button_flattr_login.set_label(_('Sign out'))
self.checkbutton_flattr_on_play.set_active(self._config.flattr.flattr_on_play)
def on_button_flattr_login(self, widget):
if not self._config.flattr.token:
try:
import webkit
except ImportError, ie:
self.show_message(_('Flattr integration requires WebKit/Gtk.'),
_('WebKit/Gtk not found'), important=True)
self.main_window.destroy()
return
gPodderFlattrSignIn(self.parent_window,
_config=self._config,
flattr=self.flattr,
set_flattr_preferences=self.set_flattr_preferences)
else:
self._config.flattr.token = ''
self.set_flattr_preferences()
def on_check_flattr_on_play(self, widget):
self._config.flattr.flattr_on_play = widget.get_active()
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)
elif container.error is not None:
self.show_message(container.error.message,
_('Extension cannot be activated'), important=True)
model.set_value(it, self.C_TOGGLE, False)
def on_treeview_button_press_event(self, treeview, event):
if event.button != 3:
return
x = int(event.x)
y = int(event.y)
path, _, _, _ = treeview.get_path_at_pos(x, y)
model = treeview.get_model()
container = model.get_value(model.get_iter(path), self.C_EXTENSION)
self.show_extension_info(model, container)
def show_extension_info(self, model, container):
if not container or not model:
return
# This is one ugly hack, but it displays the attributes of
# the metadata object of the container..
info = '\n'.join('<b>%s:</b> %s' %
tuple(map(cgi.escape, map(str, (key, value))))
for key, value in container.metadata.get_sorted())
self.show_message(info, _('Extension module info'), important=True)
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 = 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.videoplayer = self.video_player_model.get_command(index)
def on_combobox_preferred_video_format_changed(self, widget):
index = self.combobox_preferred_video_format.get_active()
self.preferred_video_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)
if result:
self._config.player = result
index = self.audio_player_model.get_index(self._config.player)
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.videoplayer)
if result:
self._config.videoplayer = result
index = self.video_player_model.get_index(self._config.videoplayer)
self.combo_video_player_app.set_active(index)
def format_update_interval_value(self, scale, value):
value = int(value)
if value == 0:
return _('manually')
2010-11-23 12:29:18 +01:00
elif value > 0 and len(self.update_interval_presets) > value:
return util.format_seconds_to_hour_min_sec(self.update_interval_presets[value]*60)
else:
return str(value)
def on_update_interval_value_changed(self, range):
value = int(range.get_value())
self._config.auto_update_feeds = (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_remove_played_episodes = False
self._config.auto_remove_unplayed_episodes = False
else:
self._config.auto_remove_played_episodes = True
self._config.episode_old_age = 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_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)
2013-02-01 04:36:21 +01:00
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)
2013-02-01 04:36:21 +01:00
def toggle_playlist_interface(self,enabled):
if (enabled==True and not(self._config.device_sync.device_type == 'none')):
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_alignment(0., .5)
else:
self.btn_playlistfolder.set_sensitive(False)
self.btn_playlistfolder.set_label('')
self.checkbutton_delete_using_playlists.set_sensitive(False)
2013-02-01 04:36:21 +01:00
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)
2013-02-01 04:36:21 +01:00
self.checkbutton_create_playlists.set_sensitive(False)
self.toggle_playlist_interface(False)
elif device_type == 'filesystem':
self.btn_filesystemMountpoint.set_label(self._config.device_sync.device_folder)
self.btn_filesystemMountpoint.set_sensitive(True)
2013-02-01 04:36:21 +01:00
self.checkbutton_create_playlists.set_sensitive(True)
children = self.btn_filesystemMountpoint.get_children()
if children:
label = children.pop()
label.set_alignment(0., .5)
self.toggle_playlist_interface(self._config.device_sync.playlists.create)
else:
# TODO: Add support for iPod and MTP devices
pass
def on_btn_device_mountpoint_clicked(self, widget):
fs = gtk.FileChooserDialog(title=_('Select folder for mount point'),
action=gtk.FILE_CHOOSER_ACTION_SELECT_FOLDER)
fs.add_button(gtk.STOCK_CANCEL, gtk.RESPONSE_CANCEL)
fs.add_button(gtk.STOCK_OPEN, gtk.RESPONSE_OK)
fs.set_current_folder(self.btn_filesystemMountpoint.get_label())
if fs.run() == gtk.RESPONSE_OK:
filename = fs.get_filename()
if self._config.device_sync.device_type == 'filesystem':
self._config.device_sync.device_folder = filename
# Request an update of the mountpoint button
self.on_combobox_device_type_changed(None)
fs.destroy()
2013-02-01 04:36:21 +01:00
def on_btn_playlist_folder_clicked(self, widget):
fs = gtk.FileChooserDialog(title=_('Select folder for playlists'),
action=gtk.FILE_CHOOSER_ACTION_SELECT_FOLDER)
fs.add_button(gtk.STOCK_CANCEL, gtk.RESPONSE_CANCEL)
fs.add_button(gtk.STOCK_OPEN, gtk.RESPONSE_OK)
fs.set_current_folder(self.btn_playlistfolder.get_label())
if fs.run() == gtk.RESPONSE_OK:
filename = util.relpath(self._config.device_sync.device_folder,
fs.get_filename())
if self._config.device_sync.device_type == 'filesystem':
self._config.device_sync.playlists.folder = filename
self.btn_playlistfolder.set_label(filename)
children = self.btn_playlistfolder.get_children()
if children:
label = children.pop()
label.set_alignment(0., .5)
fs.destroy()