# -*- 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 .
#
import os
import platform
import gi
gi.require_version('Gtk', '3.0')
from gi.repository import Gio
from gi.repository import GLib
from gi.repository import Gtk
from gi.repository import Gdk
from gi.repository import GdkPixbuf
from gi.repository import GObject
from gi.repository import Pango
import random
import re
import sys
import shutil
import subprocess
import glob
import time
import threading
import tempfile
import collections
import urllib.request, urllib.parse, urllib.error
import cgi
import gpodder
import dbus
import dbus.service
import dbus.mainloop
import dbus.glib
from gpodder import core
from gpodder import feedcore
from gpodder import util
from gpodder import opml
from gpodder import download
from gpodder import my
from gpodder import youtube
from gpodder import player
from gpodder import common
import logging
logger = logging.getLogger(__name__)
_ = gpodder.gettext
N_ = gpodder.ngettext
from gpodder.gtkui.model import Model
from gpodder.gtkui.model import PodcastListModel
from gpodder.gtkui.model import EpisodeListModel
from gpodder.gtkui.config import UIConfig
from gpodder.gtkui.services import CoverDownloader
from gpodder.gtkui.widgets import SimpleMessageArea
from gpodder.gtkui.desktopfile import UserAppsReader
from gpodder.gtkui.draw import draw_text_box_centered, draw_cake_pixbuf
from gpodder.gtkui.draw import EPISODE_LIST_ICON_SIZE
from gpodder.gtkui.interface.common import BuilderWidget
from gpodder.gtkui.interface.common import TreeViewHelper
from gpodder.gtkui.interface.addpodcast import gPodderAddPodcast
from gpodder.gtkui.download import DownloadStatusModel
from gpodder.gtkui.desktop.welcome import gPodderWelcome
from gpodder.gtkui.desktop.channel import gPodderChannel
from gpodder.gtkui.desktop.preferences import gPodderPreferences
from gpodder.gtkui.desktop.episodeselector import gPodderEpisodeSelector
from gpodder.gtkui.desktop.podcastdirectory import gPodderPodcastDirectory
from gpodder.gtkui.interface.progress import ProgressIndicator
from gpodder.gtkui.desktop.sync import gPodderSyncUI
from gpodder.gtkui import shownotes
from gpodder.dbusproxy import DBusPodcastsProxy
from gpodder import extensions
class gPodder(BuilderWidget, dbus.service.Object):
# Width (in pixels) of episode list icon
EPISODE_LIST_ICON_WIDTH = 40
def __init__(self, app, bus_name, gpodder_core, options):
dbus.service.Object.__init__(self, object_path=gpodder.dbus_gui_object_path, bus_name=bus_name)
self.podcasts_proxy = DBusPodcastsProxy(lambda: self.channels,
self.on_itemUpdate_activate,
self.playback_episodes,
self.download_episode_list,
self.episode_object_by_uri,
bus_name)
self.application = app
self.core = gpodder_core
self.config = self.core.config
self.db = self.core.db
self.model = self.core.model
self.options = options
self.extensions_menu = None
self.extensions_actions = []
BuilderWidget.__init__(self, None, _builder_expose={'app': app})
def new(self):
gpodder.user_extensions.on_ui_object_available('gpodder-gtk', self)
self.toolbar.set_property('visible', self.config.show_toolbar)
self.bluetooth_available = util.bluetooth_available()
self.config.connect_gtk_window(self.main_window, 'main_window')
self.config.connect_gtk_paned('ui.gtk.state.main_window.paned_position', self.channelPaned)
self.main_window.show()
self.player_receiver = player.MediaPlayerDBusReceiver(self.on_played)
self.gPodder.connect('key-press-event', self.on_key_press)
self.episode_columns_menu = None
self.config.add_observer(self.on_config_changed)
self.shownotes_pane = Gtk.Box()
self.shownotes_object = shownotes.get_shownotes(self.config.ui.gtk.html_shownotes, self.shownotes_pane)
# Vertical paned for the episode list and shownotes
self.vpaned = Gtk.Paned(orientation=Gtk.Orientation.VERTICAL)
paned = self.vbox_episode_list.get_parent()
self.vbox_episode_list.reparent(self.vpaned)
self.vpaned.child_set_property(self.vbox_episode_list, 'resize', True)
self.vpaned.child_set_property(self.vbox_episode_list, 'shrink', False)
self.vpaned.pack2(self.shownotes_pane, resize=False, shrink=False)
self.vpaned.show()
# Minimum height for both episode list and shownotes
self.vbox_episode_list.set_size_request(-1, 100)
self.shownotes_pane.set_size_request(-1, 100)
self.config.connect_gtk_paned('ui.gtk.state.main_window.episode_list_size',
self.vpaned)
paned.add2(self.vpaned)
self.new_episodes_window = None
self.download_status_model = DownloadStatusModel()
self.download_queue_manager = download.DownloadQueueManager(self.config, self.download_status_model)
self.config.connect_gtk_spinbutton('max_downloads', self.spinMaxDownloads)
self.config.connect_gtk_togglebutton('max_downloads_enabled', self.cbMaxDownloads)
self.config.connect_gtk_spinbutton('limit_rate_value', self.spinLimitDownloads)
self.config.connect_gtk_togglebutton('limit_rate', self.cbLimitDownloads)
# When the amount of maximum downloads changes, notify the queue manager
changed_cb = lambda spinbutton: self.download_queue_manager.update_max_downloads()
self.spinMaxDownloads.connect('value-changed', changed_cb)
# Keep a reference to the last add podcast dialog instance
self._add_podcast_dialog = None
self.default_title = None
self.set_title(_('gPodder'))
self.cover_downloader = CoverDownloader()
# Generate list models for podcasts and their episodes
self.podcast_list_model = PodcastListModel(self.cover_downloader)
self.apply_podcast_list_hide_boring()
self.cover_downloader.register('cover-available', self.cover_download_finished)
# Source IDs for timeouts for search-as-you-type
self._podcast_list_search_timeout = None
self._episode_list_search_timeout = None
# Subscribed channels
self.active_channel = None
self.channels = self.model.get_podcasts()
# For loading the list model
self.episode_list_model = EpisodeListModel(self.config, self.on_episode_list_filter_changed)
self.create_actions()
# Init the treeviews that we use
self.init_podcast_list_treeview()
self.init_episode_list_treeview()
self.init_download_list_treeview()
self.download_tasks_seen = set()
self.download_list_update_enabled = False
self.download_task_monitors = set()
# Set up the first instance of MygPoClient
self.mygpo_client = my.MygPoClient(self.config)
self.inject_extensions_menu()
gpodder.user_extensions.on_ui_initialized(self.model,
self.extensions_podcast_update_cb,
self.extensions_episode_download_cb)
gpodder.user_extensions.on_application_started()
# load list of user applications for audio playback
self.user_apps_reader = UserAppsReader(['audio', 'video'])
util.run_in_background(self.user_apps_reader.read)
# Now, update the feed cache, when everything's in place
self.btnUpdateFeeds.show()
self.feed_cache_update_cancelled = False
self.update_podcast_list_model()
self.message_area = None
self.partial_downloads_indicator = None
util.run_in_background(self.find_partial_downloads)
# Start the auto-update procedure
self._auto_update_timer_source_id = None
if self.config.auto_update_feeds:
self.restart_auto_update_timer()
# Find expired (old) episodes and delete them
old_episodes = list(common.get_expired_episodes(self.channels, self.config))
if len(old_episodes) > 0:
self.delete_episode_list(old_episodes, confirm=False)
updated_urls = set(e.channel.url for e in old_episodes)
self.update_podcast_list_model(updated_urls)
# Do the initial sync with the web service
if self.mygpo_client.can_access_webservice():
util.idle_add(self.mygpo_client.flush, True)
# First-time users should be asked if they want to see the OPML
if self.options.subscribe:
util.idle_add(self.subscribe_to_url, self.options.subscribe)
elif not self.channels:
self.on_itemUpdate_activate()
elif self.config.software_update.check_on_startup:
# Check for software updates from gpodder.org
diff = time.time() - self.config.software_update.last_check
if diff > (60*60*24)*self.config.software_update.interval:
self.config.software_update.last_check = int(time.time())
self.check_for_updates(silent=True)
def create_actions(self):
g = self.gPodder
action = Gio.SimpleAction.new_stateful(
'showEpisodeDescription', None, GLib.Variant.new_boolean(self.config.episode_list_descriptions))
action.connect('activate', self.on_itemShowDescription_activate)
g.add_action(action)
action = Gio.SimpleAction.new_stateful(
'viewHideBoringPodcasts', None, GLib.Variant.new_boolean(self.config.podcast_list_hide_boring))
action.connect('activate', self.on_item_view_hide_boring_podcasts_toggled)
g.add_action(action)
value = EpisodeListModel.VIEWS[self.config.episode_list_view_mode or EpisodeListModel.VIEW_ALL]
action = Gio.SimpleAction.new_stateful(
'viewEpisodes', GLib.VariantType.new('s'), GLib.Variant.new_string(value))
action.connect('activate', self.on_item_view_episodes_changed)
g.add_action(action)
action = Gio.SimpleAction.new('update', None)
action.connect('activate', self.on_itemUpdate_activate)
g.add_action(action)
self.update_action = action
action = Gio.SimpleAction.new('downloadAllNew', None)
action.connect('activate', self.on_itemDownloadAllNew_activate)
g.add_action(action)
action = Gio.SimpleAction.new('removeOldEpisodes', None)
action.connect('activate', self.on_itemRemoveOldEpisodes_activate)
g.add_action(action)
action = Gio.SimpleAction.new('discover', None)
action.connect('activate', self.on_itemImportChannels_activate)
g.add_action(action)
action = Gio.SimpleAction.new('addChannel', None)
action.connect('activate', self.on_itemAddChannel_activate)
g.add_action(action)
action = Gio.SimpleAction.new('massUnsubscribe', None)
action.connect('activate', self.on_itemMassUnsubscribe_activate)
g.add_action(action)
action = Gio.SimpleAction.new('updateChannel', None)
action.connect('activate', self.on_itemUpdateChannel_activate)
g.add_action(action)
self.update_channel_action = action
action = Gio.SimpleAction.new('editChannel', None)
action.connect('activate', self.on_itemEditChannel_activate)
g.add_action(action)
self.edit_channel_action = action
action = Gio.SimpleAction.new('importFromFile', None)
action.connect('activate', self.on_item_import_from_file_activate)
g.add_action(action)
action = Gio.SimpleAction.new('exportChannels', None)
action.connect('activate', self.on_itemExportChannels_activate)
g.add_action(action)
action = Gio.SimpleAction.new('play', None)
action.connect('activate', self.on_playback_selected_episodes)
g.add_action(action)
self.play_action = action
action = Gio.SimpleAction.new('open', None)
action.connect('activate', self.on_playback_selected_episodes)
g.add_action(action)
self.open_action = action
action = Gio.SimpleAction.new('download', None)
action.connect('activate', self.on_download_selected_episodes)
g.add_action(action)
self.download_action = action
action = Gio.SimpleAction.new('cancel', None)
action.connect('activate', self.on_item_cancel_download_activate)
g.add_action(action)
self.cancel_action = action
action = Gio.SimpleAction.new('delete', None)
action.connect('activate', self.on_btnDownloadedDelete_clicked)
g.add_action(action)
self.delete_action = action
action = Gio.SimpleAction.new('toggleEpisodeNew', None)
action.connect('activate', self.on_item_toggle_played_activate)
g.add_action(action)
self.toggle_episode_new_action = action
action = Gio.SimpleAction.new('toggleEpisodeLock', None)
action.connect('activate', self.on_item_toggle_lock_activate)
g.add_action(action)
self.toggle_episode_lock_action = action
action = Gio.SimpleAction.new('toggleShownotes', None)
action.connect('activate', self.on_shownotes_selected_episodes)
g.add_action(action)
action = Gio.SimpleAction.new_stateful(
'showToolbar', None, GLib.Variant.new_boolean(self.config.show_toolbar))
action.connect('activate', self.on_itemShowToolbar_activate)
g.add_action(action)
action = Gio.SimpleAction.new('sync', None)
action.connect('activate', self.on_sync_to_device_activate)
g.add_action(action)
action = Gio.SimpleAction.new('updateYoutubeSubscriptions', None)
action.connect('activate', self.on_update_youtube_subscriptions_activate)
g.add_action(action)
def inject_extensions_menu(self):
"""
Update Extras/Extensions menu.
Called at startup and when en/dis-abling extenstions.
"""
def gen_callback(label, callback):
return lambda action, param: callback()
for a in self.extensions_actions:
self.gPodder.remove_action(a.get_property('name'))
self.extensions_actions = []
if self.extensions_menu is None:
# insert menu section at startup (hides when empty)
self.extensions_menu = Gio.Menu.new()
menubar = self.application.get_menubar()
for i in range(0, menubar.get_n_items()):
menu = menubar.do_get_item_link(menubar, i, Gio.MENU_LINK_SUBMENU)
menuname = menubar.get_item_attribute_value(i, Gio.MENU_ATTRIBUTE_LABEL, None)
if menuname is not None and menuname.get_string() == 'E_xtras':
menu.append_section(_('Extensions'), self.extensions_menu)
else:
self.extensions_menu.remove_all()
extension_entries = gpodder.user_extensions.on_create_menu()
if extension_entries:
# populate menu
for i, (label, callback) in enumerate(extension_entries):
action_id = 'extensions.action_%d' % i
action = Gio.SimpleAction.new(action_id)
action.connect('activate', gen_callback(label, callback))
self.extensions_actions.append(action)
self.gPodder.add_action(action)
itm = Gio.MenuItem.new(label, 'win.' + action_id)
self.extensions_menu.append_item(itm)
def find_partial_downloads(self):
def start_progress_callback(count):
self.partial_downloads_indicator = ProgressIndicator(
_('Loading incomplete downloads'),
_('Some episodes have not finished downloading in a previous session.'),
False, self.get_dialog_parent())
self.partial_downloads_indicator.on_message(N_('%(count)d partial file', '%(count)d partial files', count) % {'count':count})
util.idle_add(self.wNotebook.set_current_page, 1)
def progress_callback(title, progress):
self.partial_downloads_indicator.on_message(title)
self.partial_downloads_indicator.on_progress(progress)
def finish_progress_callback(resumable_episodes):
util.idle_add(self.partial_downloads_indicator.on_finished)
self.partial_downloads_indicator = None
if resumable_episodes:
def offer_resuming():
self.download_episode_list_paused(resumable_episodes)
resume_all = Gtk.Button(_('Resume all'))
def on_resume_all(button):
selection = self.treeDownloads.get_selection()
selection.select_all()
selected_tasks, _, _, _, _, _ = self.downloads_list_get_selection()
selection.unselect_all()
self._for_each_task_set_status(selected_tasks, download.DownloadTask.QUEUED)
self.message_area.hide()
resume_all.connect('clicked', on_resume_all)
self.message_area = SimpleMessageArea(
_('Incomplete downloads from a previous session were found.'),
(resume_all,))
self.vboxDownloadStatusWidgets.attach(self.message_area, 0, -1, 1, 1)
self.message_area.show_all()
common.clean_up_downloads(delete_partial=False)
util.idle_add(offer_resuming)
else:
util.idle_add(self.wNotebook.set_current_page, 0)
common.find_partial_downloads(self.channels,
start_progress_callback,
progress_callback,
finish_progress_callback)
def episode_object_by_uri(self, uri):
"""Get an episode object given a local or remote URI
This can be used to quickly access an episode object
when all we have is its download filename or episode
URL (e.g. from external D-Bus calls / signals, etc..)
"""
if uri.startswith('/'):
uri = 'file://' + urllib.parse.quote(uri)
prefix = 'file://' + urllib.parse.quote(gpodder.downloads)
# By default, assume we can't pre-select any channel
# but can match episodes simply via the download URL
is_channel = lambda c: True
is_episode = lambda e: e.url == uri
if uri.startswith(prefix):
# File is on the local filesystem in the download folder
# Try to reduce search space by pre-selecting the channel
# based on the folder name of the local file
filename = urllib.parse.unquote(uri[len(prefix):])
file_parts = [_f for _f in filename.split(os.sep) if _f]
if len(file_parts) != 2:
return None
foldername, filename = file_parts
is_channel = lambda c: c.download_folder == foldername
is_episode = lambda e: e.download_filename == filename
# Deep search through channels and episodes for a match
for channel in filter(is_channel, self.channels):
for episode in filter(is_episode, channel.get_all_episodes()):
return episode
return None
def on_played(self, start, end, total, file_uri):
"""Handle the "played" signal from a media player"""
if start == 0 and end == 0 and total == 0:
# Ignore bogus play event
return
elif end < start + 5:
# Ignore "less than five seconds" segments,
# as they can happen with seeking, etc...
return
logger.debug('Received play action: %s (%d, %d, %d)', file_uri, start, end, total)
episode = self.episode_object_by_uri(file_uri)
if episode is not None:
file_type = episode.file_type()
now = time.time()
if total > 0:
episode.total_time = total
elif total == 0:
# Assume the episode's total time for the action
total = episode.total_time
assert (episode.current_position_updated is None or
now >= episode.current_position_updated)
episode.current_position = end
episode.current_position_updated = now
episode.mark(is_played=True)
episode.save()
self.episode_list_status_changed([episode])
# Submit this action to the webservice
self.mygpo_client.on_playback_full(episode, start, end, total)
def on_add_remove_podcasts_mygpo(self):
actions = self.mygpo_client.get_received_actions()
if not actions:
return False
existing_urls = [c.url for c in self.channels]
# Columns for the episode selector window - just one...
columns = (
('description', None, None, _('Action')),
)
# A list of actions that have to be chosen from
changes = []
# Actions that are ignored (already carried out)
ignored = []
for action in actions:
if action.is_add and action.url not in existing_urls:
changes.append(my.Change(action))
elif action.is_remove and action.url in existing_urls:
podcast_object = None
for podcast in self.channels:
if podcast.url == action.url:
podcast_object = podcast
break
changes.append(my.Change(action, podcast_object))
else:
ignored.append(action)
# Confirm all ignored changes
self.mygpo_client.confirm_received_actions(ignored)
def execute_podcast_actions(selected):
# In the future, we might retrieve the title from gpodder.net here,
# but for now, we just use "None" to use the feed-provided title
title = None
add_list = [(title, c.action.url)
for c in selected if c.action.is_add]
remove_list = [c.podcast for c in selected if c.action.is_remove]
# Apply the accepted changes locally
self.add_podcast_list(add_list)
self.remove_podcast_list(remove_list, confirm=False)
# All selected items are now confirmed
self.mygpo_client.confirm_received_actions(c.action for c in selected)
# Revert the changes on the server
rejected = [c.action for c in changes if c not in selected]
self.mygpo_client.reject_received_actions(rejected)
def ask():
# We're abusing the Episode Selector again ;) -- thp
gPodderEpisodeSelector(self.main_window, \
title=_('Confirm changes from gpodder.net'), \
instructions=_('Select the actions you want to carry out.'), \
episodes=changes, \
columns=columns, \
size_attribute=None, \
stock_ok_button=Gtk.STOCK_APPLY, \
callback=execute_podcast_actions, \
_config=self.config)
# There are some actions that need the user's attention
if changes:
util.idle_add(ask)
return True
# We have no remaining actions - no selection happens
return False
def rewrite_urls_mygpo(self):
# Check if we have to rewrite URLs since the last add
rewritten_urls = self.mygpo_client.get_rewritten_urls()
changed = False
for rewritten_url in rewritten_urls:
if not rewritten_url.new_url:
continue
for channel in self.channels:
if channel.url == rewritten_url.old_url:
logger.info('Updating URL of %s to %s', channel,
rewritten_url.new_url)
channel.url = rewritten_url.new_url
channel.save()
changed = True
break
if changed:
util.idle_add(self.update_episode_list_model)
def on_send_full_subscriptions(self):
# Send the full subscription list to the gpodder.net client
# (this will overwrite the subscription list on the server)
indicator = ProgressIndicator(_('Uploading subscriptions'), \
_('Your subscriptions are being uploaded to the server.'), \
False, self.get_dialog_parent())
try:
self.mygpo_client.set_subscriptions([c.url for c in self.channels])
util.idle_add(self.show_message, _('List uploaded successfully.'))
except Exception as e:
def show_error(e):
message = str(e)
if not message:
message = e.__class__.__name__
self.show_message(message, \
_('Error while uploading'), \
important=True)
util.idle_add(show_error, e)
util.idle_add(indicator.on_finished)
def on_button_subscribe_clicked(self, button):
self.on_itemImportChannels_activate(button)
def on_button_downloads_clicked(self, widget):
self.downloads_window.show()
def for_each_episode_set_task_status(self, episodes, status):
episode_urls = set(episode.url for episode in episodes)
model = self.treeDownloads.get_model()
selected_tasks = [(Gtk.TreeRowReference.new(model, row.path), \
model.get_value(row.iter, \
DownloadStatusModel.C_TASK)) for row in model \
if model.get_value(row.iter, DownloadStatusModel.C_TASK).url \
in episode_urls]
self._for_each_task_set_status(selected_tasks, status)
def on_treeview_button_pressed(self, treeview, event):
if event.window != treeview.get_bin_window():
return False
role = getattr(treeview, TreeViewHelper.ROLE)
if role == TreeViewHelper.ROLE_EPISODES and event.button == 1:
# Toggle episode "new" status by clicking the icon (bug 1432)
result = treeview.get_path_at_pos(int(event.x), int(event.y))
if result is not None:
path, column, x, y = result
# The user clicked the icon if she clicked in the first column
# and the x position is in the area where the icon resides
if (x < self.EPISODE_LIST_ICON_WIDTH and
column == treeview.get_columns()[0]):
model = treeview.get_model()
cursor_episode = model.get_value(model.get_iter(path),
EpisodeListModel.C_EPISODE)
new_value = cursor_episode.is_new
selected_episodes = self.get_selected_episodes()
# Avoid changing anything if the clicked episode is not
# selected already - otherwise update all selected
if cursor_episode in selected_episodes:
for episode in selected_episodes:
episode.mark(is_played=new_value)
self.update_episode_list_icons(selected=True)
self.update_podcast_list_model(selected=True)
return True
return event.button == 3
def on_treeview_podcasts_button_released(self, treeview, event):
if event.window != treeview.get_bin_window():
return False
return self.treeview_channels_show_context_menu(treeview, event)
def on_treeview_episodes_button_released(self, treeview, event):
if event.window != treeview.get_bin_window():
return False
return self.treeview_available_show_context_menu(treeview, event)
def on_treeview_downloads_button_released(self, treeview, event):
if event.window != treeview.get_bin_window():
return False
return self.treeview_downloads_show_context_menu(treeview, event)
def on_entry_search_podcasts_changed(self, editable):
if self.hbox_search_podcasts.get_property('visible'):
def set_search_term(self, text):
self.podcast_list_model.set_search_term(text)
self._podcast_list_search_timeout = None
return False
if self._podcast_list_search_timeout is not None:
GObject.source_remove(self._podcast_list_search_timeout)
self._podcast_list_search_timeout = GObject.timeout_add(
self.config.ui.gtk.live_search_delay,
set_search_term, self, editable.get_chars(0, -1))
def on_entry_search_podcasts_key_press(self, editable, event):
if event.keyval == Gdk.KEY_Escape:
self.hide_podcast_search()
return True
def hide_podcast_search(self, *args):
if self._podcast_list_search_timeout is not None:
GObject.source_remove(self._podcast_list_search_timeout)
self._podcast_list_search_timeout = None
self.hbox_search_podcasts.hide()
self.entry_search_podcasts.set_text('')
self.podcast_list_model.set_search_term(None)
self.treeChannels.grab_focus()
def show_podcast_search(self, input_char):
self.hbox_search_podcasts.show()
self.entry_search_podcasts.insert_text(input_char, -1)
self.entry_search_podcasts.grab_focus()
self.entry_search_podcasts.set_position(-1)
def init_podcast_list_treeview(self):
# Set up podcast channel tree view widget
column = Gtk.TreeViewColumn('')
iconcell = Gtk.CellRendererPixbuf()
iconcell.set_property('width', 45)
column.pack_start(iconcell, False)
column.add_attribute(iconcell, 'pixbuf', PodcastListModel.C_COVER)
column.add_attribute(iconcell, 'visible', PodcastListModel.C_COVER_VISIBLE)
namecell = Gtk.CellRendererText()
namecell.set_property('ellipsize', Pango.EllipsizeMode.END)
column.pack_start(namecell, True)
column.add_attribute(namecell, 'markup', PodcastListModel.C_DESCRIPTION)
iconcell = Gtk.CellRendererPixbuf()
iconcell.set_property('xalign', 1.0)
column.pack_start(iconcell, False)
column.add_attribute(iconcell, 'pixbuf', PodcastListModel.C_PILL)
column.add_attribute(iconcell, 'visible', PodcastListModel.C_PILL_VISIBLE)
self.treeChannels.append_column(column)
self.treeChannels.set_model(self.podcast_list_model.get_filtered_model())
# When no podcast is selected, clear the episode list model
selection = self.treeChannels.get_selection()
def select_function(selection, model, path, path_currently_selected):
url = model.get_value(model.get_iter(path), PodcastListModel.C_URL)
return (url != '-')
selection.set_select_function(select_function)#, full=True)
# Set up type-ahead find for the podcast list
def on_key_press(treeview, event):
if event.keyval == Gdk.KEY_Right:
self.treeAvailable.grab_focus()
elif event.keyval in (Gdk.KEY_Up, Gdk.KEY_Down):
# If section markers exist in the treeview, we want to
# "jump over" them when moving the cursor up and down
selection = self.treeChannels.get_selection()
model, it = selection.get_selected()
if event.keyval == Gdk.KEY_Up:
step = -1
else:
step = 1
path = model.get_path(it)
while True:
path = (path[0]+step,)
if path[0] < 0:
# Valid paths must have a value >= 0
return True
try:
it = model.get_iter(path)
except ValueError:
# Already at the end of the list
return True
if model.get_value(it, PodcastListModel.C_URL) != '-':
break
self.treeChannels.set_cursor(path)
elif event.keyval == Gdk.KEY_Escape:
self.hide_podcast_search()
elif event.get_state() & Gdk.ModifierType.CONTROL_MASK:
# Don't handle type-ahead when control is pressed (so shortcuts
# with the Ctrl key still work, e.g. Ctrl+A, ...)
return True
elif event.keyval == Gdk.KEY_Delete:
return False
else:
unicode_char_id = Gdk.keyval_to_unicode(event.keyval)
# < 32 to intercept Delete and Tab events
if unicode_char_id < 32:
return False
input_char = chr(unicode_char_id)
self.show_podcast_search(input_char)
return True
self.treeChannels.connect('key-press-event', on_key_press)
self.treeChannels.connect('popup-menu', self.treeview_channels_show_context_menu)
# Enable separators to the podcast list to separate special podcasts
# from others (this is used for the "all episodes" view)
self.treeChannels.set_row_separator_func(PodcastListModel.row_separator_func)
TreeViewHelper.set(self.treeChannels, TreeViewHelper.ROLE_PODCASTS)
def on_entry_search_episodes_changed(self, editable):
if self.hbox_search_episodes.get_property('visible'):
def set_search_term(self, text):
self.episode_list_model.set_search_term(text)
self._episode_list_search_timeout = None
return False
if self._episode_list_search_timeout is not None:
GObject.source_remove(self._episode_list_search_timeout)
self._episode_list_search_timeout = GObject.timeout_add(
self.config.ui.gtk.live_search_delay,
set_search_term, self, editable.get_chars(0, -1))
def on_entry_search_episodes_key_press(self, editable, event):
if event.keyval == Gdk.KEY_Escape:
self.hide_episode_search()
return True
def hide_episode_search(self, *args):
if self._episode_list_search_timeout is not None:
GObject.source_remove(self._episode_list_search_timeout)
self._episode_list_search_timeout = None
self.hbox_search_episodes.hide()
self.entry_search_episodes.set_text('')
self.episode_list_model.set_search_term(None)
self.treeAvailable.grab_focus()
def show_episode_search(self, input_char):
self.hbox_search_episodes.show()
self.entry_search_episodes.insert_text(input_char, -1)
self.entry_search_episodes.grab_focus()
self.entry_search_episodes.set_position(-1)
def set_episode_list_column(self, index, new_value):
mask = (1 << index)
if new_value:
self.config.episode_list_columns |= mask
else:
self.config.episode_list_columns &= ~mask
def update_episode_list_columns_visibility(self):
columns = TreeViewHelper.get_columns(self.treeAvailable)
for index, column in enumerate(columns):
visible = bool(self.config.episode_list_columns & (1 << index))
column.set_visible(visible)
self.view_column_actions[index].set_state(GLib.Variant.new_boolean(visible))
self.treeAvailable.columns_autosize()
def on_episode_list_header_clicked(self, button, event):
if event.button != 3:
return False
if self.episode_columns_menu is not None:
self.episode_columns_menu.popup(None, None, None, None, event.button, event.time)
return False
def init_episode_list_treeview(self):
self.episode_list_model.set_view_mode(self.config.episode_list_view_mode)
self.treeAvailable.set_model(self.episode_list_model.get_filtered_model())
TreeViewHelper.set(self.treeAvailable, TreeViewHelper.ROLE_EPISODES)
iconcell = Gtk.CellRendererPixbuf()
episode_list_icon_size = Gtk.icon_size_register('episode-list',
EPISODE_LIST_ICON_SIZE, EPISODE_LIST_ICON_SIZE)
iconcell.set_property('stock-size', episode_list_icon_size)
iconcell.set_fixed_size(self.EPISODE_LIST_ICON_WIDTH, -1)
namecell = Gtk.CellRendererText()
namecell.set_property('ellipsize', Pango.EllipsizeMode.END)
namecolumn = Gtk.TreeViewColumn(_('Episode'))
namecolumn.pack_start(iconcell, False)
namecolumn.add_attribute(iconcell, 'icon-name', EpisodeListModel.C_STATUS_ICON)
namecolumn.pack_start(namecell, True)
namecolumn.add_attribute(namecell, 'markup', EpisodeListModel.C_DESCRIPTION)
namecolumn.set_sort_column_id(EpisodeListModel.C_DESCRIPTION)
namecolumn.set_sizing(Gtk.TreeViewColumnSizing.AUTOSIZE)
namecolumn.set_resizable(True)
namecolumn.set_expand(True)
lockcell = Gtk.CellRendererPixbuf()
lockcell.set_fixed_size(40, -1)
lockcell.set_property('stock-size', Gtk.IconSize.MENU)
lockcell.set_property('icon-name', 'emblem-readonly')
namecolumn.pack_start(lockcell, False)
namecolumn.add_attribute(lockcell, 'visible', EpisodeListModel.C_LOCKED)
sizecell = Gtk.CellRendererText()
sizecell.set_property('xalign', 1)
sizecolumn = Gtk.TreeViewColumn(_('Size'), sizecell, text=EpisodeListModel.C_FILESIZE_TEXT)
sizecolumn.set_sort_column_id(EpisodeListModel.C_FILESIZE)
timecell = Gtk.CellRendererText()
timecell.set_property('xalign', 1)
timecolumn = Gtk.TreeViewColumn(_('Duration'), timecell, text=EpisodeListModel.C_TIME)
timecolumn.set_sort_column_id(EpisodeListModel.C_TOTAL_TIME)
releasecell = Gtk.CellRendererText()
releasecolumn = Gtk.TreeViewColumn(_('Released'), releasecell, text=EpisodeListModel.C_PUBLISHED_TEXT)
releasecolumn.set_sort_column_id(EpisodeListModel.C_PUBLISHED)
namecolumn.set_reorderable(True)
self.treeAvailable.append_column(namecolumn)
for itemcolumn in (sizecolumn, timecolumn, releasecolumn):
itemcolumn.set_reorderable(True)
self.treeAvailable.append_column(itemcolumn)
TreeViewHelper.register_column(self.treeAvailable, itemcolumn)
# Add context menu to all tree view column headers
for column in self.treeAvailable.get_columns():
label = Gtk.Label(label=column.get_title())
label.show_all()
column.set_widget(label)
w = column.get_widget()
while w is not None and not isinstance(w, Gtk.Button):
w = w.get_parent()
w.connect('button-release-event', self.on_episode_list_header_clicked)
# For each column that can be shown/hidden, add a menu item
self.view_column_actions = []
columns = TreeViewHelper.get_columns(self.treeAvailable)
def on_visible_toggled(action, param, index):
state = action.get_state()
self.set_episode_list_column(index, not state)
action.set_state(GLib.Variant.new_boolean(not state))
for index, column in enumerate(columns):
name = 'showColumn%i' % index
action = Gio.SimpleAction.new_stateful(
name, None, GLib.Variant.new_boolean(False))
action.connect('activate', on_visible_toggled, index)
self.main_window.add_action(action)
self.view_column_actions.append(action)
self.application.menu_view_columns.insert(index, column.get_title(), 'win.' + name)
self.episode_columns_menu = Gtk.Menu.new_from_model(self.application.menu_view_columns)
self.episode_columns_menu.attach_to_widget(self.main_window)
# Update the visibility of the columns and the check menu items
self.update_episode_list_columns_visibility()
# Set up type-ahead find for the episode list
def on_key_press(treeview, event):
if event.keyval == Gdk.KEY_Left:
self.treeChannels.grab_focus()
elif event.keyval == Gdk.KEY_Escape:
if self.hbox_search_episodes.get_property('visible'):
self.hide_episode_search()
else:
self.shownotes_object.hide_pane()
elif event.get_state() & Gdk.ModifierType.CONTROL_MASK:
# Don't handle type-ahead when control is pressed (so shortcuts
# with the Ctrl key still work, e.g. Ctrl+A, ...)
return False
else:
unicode_char_id = Gdk.keyval_to_unicode(event.keyval)
# < 32 to intercept Delete and Tab events
if unicode_char_id < 32:
return False
input_char = chr(unicode_char_id)
self.show_episode_search(input_char)
return True
self.treeAvailable.connect('key-press-event', on_key_press)
self.treeAvailable.connect('popup-menu', self.treeview_available_show_context_menu)
self.treeAvailable.enable_model_drag_source(Gdk.ModifierType.BUTTON1_MASK, \
(('text/uri-list', 0, 0),), Gdk.DragAction.COPY)
def drag_data_get(tree, context, selection_data, info, timestamp):
uris = ['file://'+e.local_filename(create=False) \
for e in self.get_selected_episodes() \
if e.was_downloaded(and_exists=True)]
uris.append('') # for the trailing '\r\n'
selection_data.set(selection_data.target, 8, '\r\n'.join(uris))
self.treeAvailable.connect('drag-data-get', drag_data_get)
selection = self.treeAvailable.get_selection()
selection.set_mode(Gtk.SelectionMode.MULTIPLE)
selection.connect('changed', self.on_episode_list_selection_changed)
def on_episode_list_selection_changed(self, selection):
# Update the toolbar buttons
self.play_or_download()
# and the shownotes
self.shownotes_object.set_episodes(self.get_selected_episodes())
def init_download_list_treeview(self):
# enable multiple selection support
self.treeDownloads.get_selection().set_mode(Gtk.SelectionMode.MULTIPLE)
self.treeDownloads.set_search_equal_func(TreeViewHelper.make_search_equal_func(DownloadStatusModel))
# columns and renderers for "download progress" tab
# First column: [ICON] Episodename
column = Gtk.TreeViewColumn(_('Episode'))
cell = Gtk.CellRendererPixbuf()
cell.set_property('stock-size', Gtk.IconSize.BUTTON)
column.pack_start(cell, False)
column.add_attribute(cell, 'icon-name', \
DownloadStatusModel.C_ICON_NAME)
cell = Gtk.CellRendererText()
cell.set_property('ellipsize', Pango.EllipsizeMode.END)
column.pack_start(cell, True)
column.add_attribute(cell, 'markup', DownloadStatusModel.C_NAME)
column.set_sizing(Gtk.TreeViewColumnSizing.AUTOSIZE)
column.set_expand(True)
self.treeDownloads.append_column(column)
# Second column: Progress
cell = Gtk.CellRendererProgress()
cell.set_property('yalign', .5)
cell.set_property('ypad', 6)
column = Gtk.TreeViewColumn(_('Progress'), cell,
value=DownloadStatusModel.C_PROGRESS, \
text=DownloadStatusModel.C_PROGRESS_TEXT)
column.set_sizing(Gtk.TreeViewColumnSizing.AUTOSIZE)
column.set_expand(False)
self.treeDownloads.append_column(column)
column.set_property('min-width', 150)
column.set_property('max-width', 150)
self.treeDownloads.set_model(self.download_status_model)
TreeViewHelper.set(self.treeDownloads, TreeViewHelper.ROLE_DOWNLOADS)
self.treeDownloads.connect('popup-menu', self.treeview_downloads_show_context_menu)
def on_treeview_expose_event(self, treeview, ctx):
model = treeview.get_model()
if (model is not None and model.get_iter_first() is not None):
return False
role = getattr(treeview, TreeViewHelper.ROLE, None)
if role is None:
return False
width = treeview.get_allocated_width()
height= treeview.get_allocated_height()
if role == TreeViewHelper.ROLE_EPISODES:
if self.config.episode_list_view_mode != EpisodeListModel.VIEW_ALL:
text = _('No episodes in current view')
else:
text = _('No episodes available')
elif role == TreeViewHelper.ROLE_PODCASTS:
if self.config.episode_list_view_mode != \
EpisodeListModel.VIEW_ALL and \
self.config.podcast_list_hide_boring and \
len(self.channels) > 0:
text = _('No podcasts in this view')
else:
text = _('No subscriptions')
elif role == TreeViewHelper.ROLE_DOWNLOADS:
text = _('No active tasks')
else:
raise Exception('on_treeview_expose_event: unknown role')
draw_text_box_centered(ctx, treeview, width, height, text, None, None)
return True
def enable_download_list_update(self):
if not self.download_list_update_enabled:
self.update_downloads_list()
GObject.timeout_add(1500, self.update_downloads_list)
self.download_list_update_enabled = True
def cleanup_downloads(self):
model = self.download_status_model
all_tasks = [(Gtk.TreeRowReference.new(model, row.path), row[0]) for row in model]
changed_episode_urls = set()
for row_reference, task in all_tasks:
if task.status in (task.DONE, task.CANCELLED):
model.remove(model.get_iter(row_reference.get_path()))
try:
# We don't "see" this task anymore - remove it;
# this is needed, so update_episode_list_icons()
# below gets the correct list of "seen" tasks
self.download_tasks_seen.remove(task)
except KeyError as key_error:
pass
changed_episode_urls.add(task.url)
# Tell the task that it has been removed (so it can clean up)
task.removed_from_list()
# Tell the podcasts tab to update icons for our removed podcasts
self.update_episode_list_icons(changed_episode_urls)
# Update the downloads list one more time
self.update_downloads_list(can_call_cleanup=False)
def on_tool_downloads_toggled(self, toolbutton):
if toolbutton.get_active():
self.wNotebook.set_current_page(1)
else:
self.wNotebook.set_current_page(0)
def add_download_task_monitor(self, monitor):
self.download_task_monitors.add(monitor)
model = self.download_status_model
if model is None:
model = ()
for row in model:
task = row[self.download_status_model.C_TASK]
monitor.task_updated(task)
def remove_download_task_monitor(self, monitor):
self.download_task_monitors.remove(monitor)
def set_download_progress(self, progress):
gpodder.user_extensions.on_download_progress(progress)
def update_downloads_list(self, can_call_cleanup=True):
try:
model = self.download_status_model
downloading, synchronizing, failed, finished, queued, paused, others = 0, 0, 0, 0, 0, 0, 0
total_speed, total_size, done_size = 0, 0, 0
# Keep a list of all download tasks that we've seen
download_tasks_seen = set()
# Do not go through the list of the model is not (yet) available
if model is None:
model = ()
for row in model:
self.download_status_model.request_update(row.iter)
task = row[self.download_status_model.C_TASK]
speed, size, status, progress, activity = task.speed, task.total_size, task.status, task.progress, task.activity
# Let the download task monitors know of changes
for monitor in self.download_task_monitors:
monitor.task_updated(task)
total_size += size
done_size += size*progress
download_tasks_seen.add(task)
if (status == download.DownloadTask.DOWNLOADING and
activity == download.DownloadTask.ACTIVITY_DOWNLOAD):
downloading += 1
total_speed += speed
elif (status == download.DownloadTask.DOWNLOADING and
activity == download.DownloadTask.ACTIVITY_SYNCHRONIZE):
synchronizing += 1
elif status == download.DownloadTask.FAILED:
failed += 1
elif status == download.DownloadTask.DONE:
finished += 1
elif status == download.DownloadTask.QUEUED:
queued += 1
elif status == download.DownloadTask.PAUSED:
paused += 1
else:
others += 1
# Remember which tasks we have seen after this run
self.download_tasks_seen = download_tasks_seen
text = [_('Progress')]
if downloading + failed + queued + synchronizing > 0:
s = []
if downloading > 0:
s.append(N_('%(count)d active', '%(count)d active', downloading) % {'count':downloading})
if synchronizing > 0:
s.append(N_('%(count)d active', '%(count)d active', synchronizing) % {'count':synchronizing})
if failed > 0:
s.append(N_('%(count)d failed', '%(count)d failed', failed) % {'count':failed})
if queued > 0:
s.append(N_('%(count)d queued', '%(count)d queued', queued) % {'count':queued})
text.append(' (' + ', '.join(s)+')')
self.labelDownloads.set_text(''.join(text))
title = [self.default_title]
# Accessing task.status_changed has the side effect of re-setting
# the changed flag, but we only do it once here so that's okay
channel_urls = [task.podcast_url for task in
self.download_tasks_seen if task.status_changed]
episode_urls = [task.url for task in self.download_tasks_seen]
if downloading > 0:
title.append(N_('downloading %(count)d file', 'downloading %(count)d files', downloading) % {'count':downloading})
if total_size > 0:
percentage = 100.0*done_size/total_size
else:
percentage = 0.0
self.set_download_progress(percentage/100)
total_speed = util.format_filesize(total_speed)
title[1] += ' (%d%%, %s/s)' % (percentage, total_speed)
if synchronizing > 0:
title.append(N_('synchronizing %(count)d file', 'synchronizing %(count)d files', synchronizing) % {'count':synchronizing})
if queued > 0:
title.append(N_('%(queued)d task queued', '%(queued)d tasks queued', queued) % {'queued':queued})
if (downloading + synchronizing + queued)==0:
self.set_download_progress(1.)
self.downloads_finished(self.download_tasks_seen)
gpodder.user_extensions.on_all_episodes_downloaded()
logger.info('All tasks have finished.')
# Remove finished episodes
if self.config.auto_cleanup_downloads and can_call_cleanup:
self.cleanup_downloads()
# Stop updating the download list here
self.download_list_update_enabled = False
self.gPodder.set_title(' - '.join(title))
self.update_episode_list_icons(episode_urls)
self.play_or_download()
if channel_urls:
self.update_podcast_list_model(channel_urls)
return self.download_list_update_enabled
except Exception as e:
logger.error('Exception happened while updating download list.', exc_info=True)
self.show_message('%s\n\n%s' % (_('Please report this problem and restart gPodder:'), str(e)), _('Unhandled exception'), important=True)
# We return False here, so the update loop won't be called again,
# that's why we require the restart of gPodder in the message.
return False
def on_config_changed(self, *args):
util.idle_add(self._on_config_changed, *args)
def _on_config_changed(self, name, old_value, new_value):
if name == 'ui.gtk.toolbar':
self.toolbar.set_property('visible', new_value)
elif name == 'ui.gtk.episode_list.descriptions':
self.update_episode_list_model()
elif name in ('auto.update.enabled', 'auto.update.frequency'):
self.restart_auto_update_timer()
elif name in ('ui.gtk.podcast_list.all_episodes',
'ui.gtk.podcast_list.sections'):
# Force a update of the podcast list model
self.update_podcast_list_model()
elif name == 'ui.gtk.episode_list.columns':
self.update_episode_list_columns_visibility()
def on_treeview_query_tooltip(self, treeview, x, y, keyboard_tooltip, tooltip):
# With get_bin_window, we get the window that contains the rows without
# the header. The Y coordinate of this window will be the height of the
# treeview header. This is the amount we have to subtract from the
# event's Y coordinate to get the coordinate to pass to get_path_at_pos
(x_bin, y_bin) = treeview.get_bin_window().get_position()
y -= x_bin
y -= y_bin
(path, column, rx, ry) = treeview.get_path_at_pos( x, y) or (None,)*4
if not getattr(treeview, TreeViewHelper.CAN_TOOLTIP) or x > 50 or (column is not None and column != treeview.get_columns()[0]):
setattr(treeview, TreeViewHelper.LAST_TOOLTIP, None)
return False
if path is not None:
model = treeview.get_model()
iter = model.get_iter(path)
role = getattr(treeview, TreeViewHelper.ROLE)
if role == TreeViewHelper.ROLE_EPISODES:
id = model.get_value(iter, EpisodeListModel.C_URL)
elif role == TreeViewHelper.ROLE_PODCASTS:
id = model.get_value(iter, PodcastListModel.C_URL)
if id == '-':
# Section header - no tooltip here (for now at least)
return False
last_tooltip = getattr(treeview, TreeViewHelper.LAST_TOOLTIP)
if last_tooltip is not None and last_tooltip != id:
setattr(treeview, TreeViewHelper.LAST_TOOLTIP, None)
return False
setattr(treeview, TreeViewHelper.LAST_TOOLTIP, id)
if role == TreeViewHelper.ROLE_EPISODES:
description = model.get_value(iter, EpisodeListModel.C_TOOLTIP)
if description:
tooltip.set_text(description)
else:
return False
elif role == TreeViewHelper.ROLE_PODCASTS:
channel = model.get_value(iter, PodcastListModel.C_CHANNEL)
if channel is None or not hasattr(channel, 'title'):
return False
error_str = model.get_value(iter, PodcastListModel.C_ERROR)
if error_str:
error_str = _('Feedparser error: %s') % cgi.escape(error_str.strip())
error_str = '%s' % error_str
table = Gtk.Table(rows=3, columns=3)
table.set_row_spacings(5)
table.set_col_spacings(5)
table.set_border_width(5)
heading = Gtk.Label()
heading.set_alignment(0, 1)
heading.set_markup('%s\n%s' % (cgi.escape(channel.title), cgi.escape(channel.url)))
table.attach(heading, 0, 1, 0, 1)
table.attach(Gtk.HSeparator(), 0, 3, 1, 2)
if len(channel.description) < 500:
description = channel.description
else:
pos = channel.description.find('\n\n')
if pos == -1 or pos > 500:
description = channel.description[:498]+'[...]'
else:
description = channel.description[:pos]
description = Gtk.Label(label=description)
if error_str:
description.set_markup(error_str)
description.set_alignment(0, 0)
description.set_line_wrap(True)
table.attach(description, 0, 3, 2, 3)
table.show_all()
tooltip.set_custom(table)
return True
setattr(treeview, TreeViewHelper.LAST_TOOLTIP, None)
return False
def treeview_allow_tooltips(self, treeview, allow):
setattr(treeview, TreeViewHelper.CAN_TOOLTIP, allow)
def treeview_handle_context_menu_click(self, treeview, event):
if event is None:
selection = treeview.get_selection()
return selection.get_selected_rows()
x, y = int(event.x), int(event.y)
path, column, rx, ry = treeview.get_path_at_pos(x, y) or (None,)*4
selection = treeview.get_selection()
model, paths = selection.get_selected_rows()
if path is None or (path not in paths and \
event.button == 3):
# We have right-clicked, but not into the selection,
# assume we don't want to operate on the selection
paths = []
if path is not None and not paths and \
event.button == 3:
# No selection or clicked outside selection;
# select the single item where we clicked
treeview.grab_focus()
treeview.set_cursor(path, column, 0)
paths = [path]
if not paths:
# Unselect any remaining items (clicked elsewhere)
if hasattr(treeview, 'is_rubber_banding_active'):
if not treeview.is_rubber_banding_active():
selection.unselect_all()
else:
selection.unselect_all()
return model, paths
def downloads_list_get_selection(self, model=None, paths=None):
if model is None and paths is None:
selection = self.treeDownloads.get_selection()
model, paths = selection.get_selected_rows()
can_queue, can_cancel, can_pause, can_remove, can_force = (True,)*5
selected_tasks = [(Gtk.TreeRowReference.new(model, path), \
model.get_value(model.get_iter(path), \
DownloadStatusModel.C_TASK)) for path in paths]
for row_reference, task in selected_tasks:
if task.status != download.DownloadTask.QUEUED:
can_force = False
if task.status not in (download.DownloadTask.PAUSED, \
download.DownloadTask.FAILED, \
download.DownloadTask.CANCELLED):
can_queue = False
if task.status not in (download.DownloadTask.PAUSED, \
download.DownloadTask.QUEUED, \
download.DownloadTask.DOWNLOADING, \
download.DownloadTask.FAILED):
can_cancel = False
if task.status not in (download.DownloadTask.QUEUED, \
download.DownloadTask.DOWNLOADING):
can_pause = False
if task.status not in (download.DownloadTask.CANCELLED, \
download.DownloadTask.FAILED, \
download.DownloadTask.DONE):
can_remove = False
return selected_tasks, can_queue, can_cancel, can_pause, can_remove, can_force
def downloads_finished(self, download_tasks_seen):
# Separate tasks into downloads & syncs
# Since calling notify_as_finished or notify_as_failed clears the flag,
# need to iterate through downloads & syncs separately, else all sync
# tasks will have their flags cleared if we do downloads first
def filter_by_activity(activity, tasks):
return [task for task in tasks if task.activity == activity]
download_tasks = filter_by_activity(download.DownloadTask.ACTIVITY_DOWNLOAD,
download_tasks_seen)
finished_downloads = [str(task)
for task in download_tasks if task.notify_as_finished()]
failed_downloads = ['%s (%s)' % (task, task.error_message)
for task in download_tasks if task.notify_as_failed()]
sync_tasks = filter_by_activity(download.DownloadTask.ACTIVITY_SYNCHRONIZE,
download_tasks_seen)
finished_syncs = [task for task in sync_tasks if task.notify_as_finished()]
failed_syncs = [task for task in sync_tasks if task.notify_as_failed()]
# Note that 'finished_ / failed_downloads' is a list of strings
# Whereas 'finished_ / failed_syncs' is a list of SyncTask objects
if finished_downloads and failed_downloads:
message = self.format_episode_list(finished_downloads, 5)
message += '\n\n%s\n' % _('Could not download some episodes:')
message += self.format_episode_list(failed_downloads, 5)
self.show_message(message, _('Downloads finished'))
elif finished_downloads:
message = self.format_episode_list(finished_downloads)
self.show_message(message, _('Downloads finished'))
elif failed_downloads:
message = self.format_episode_list(failed_downloads)
self.show_message(message, _('Downloads failed'))
if finished_syncs and failed_syncs:
message = self.format_episode_list(list(map((lambda task: str(task)),finished_syncs)), 5)
message += '\n\n%s\n' % _('Could not sync some episodes:')
message += self.format_episode_list(list(map((lambda task: str(task)),failed_syncs)), 5)
self.show_message(message, _('Device synchronization finished'), True)
elif finished_syncs:
message = self.format_episode_list(list(map((lambda task: str(task)),finished_syncs)))
self.show_message(message, _('Device synchronization finished'))
elif failed_syncs:
message = self.format_episode_list(list(map((lambda task: str(task)),failed_syncs)))
self.show_message(message, _('Device synchronization failed'), True)
# Do post-sync processing if required
for task in finished_syncs:
if self.config.device_sync.after_sync.mark_episodes_played:
logger.info('Marking as played on transfer: %s', task.episode.url)
task.episode.mark(is_played=True)
if self.config.device_sync.after_sync.delete_episodes:
logger.info('Removing episode after transfer: %s', task.episode.url)
task.episode.delete_from_disk()
self.sync_ui.device.close()
# Update icon list to show changes, if any
self.update_episode_list_icons(all=True)
self.update_podcast_list_model()
def format_episode_list(self, episode_list, max_episodes=10):
"""
Format a list of episode names for notifications
Will truncate long episode names and limit the amount of
episodes displayed (max_episodes=10).
The episode_list parameter should be a list of strings.
"""
MAX_TITLE_LENGTH = 100
result = []
for title in episode_list[:min(len(episode_list), max_episodes)]:
# Bug 1834: make sure title is a unicode string,
# so it may be cut correctly on UTF-8 char boundaries
title = util.convert_bytes(title)
if len(title) > MAX_TITLE_LENGTH:
middle = (MAX_TITLE_LENGTH//2)-2
title = '%s...%s' % (title[0:middle], title[-middle:])
result.append(cgi.escape(title))
result.append('\n')
more_episodes = len(episode_list) - max_episodes
if more_episodes > 0:
result.append('(...')
result.append(N_('%(count)d more episode', '%(count)d more episodes', more_episodes) % {'count':more_episodes})
result.append('...)')
return (''.join(result)).strip()
def _for_each_task_set_status(self, tasks, status, force_start=False):
episode_urls = set()
model = self.treeDownloads.get_model()
for row_reference, task in tasks:
if status == download.DownloadTask.QUEUED:
# Only queue task when its paused/failed/cancelled (or forced)
if task.status in (task.PAUSED, task.FAILED, task.CANCELLED) or force_start:
if force_start:
self.download_queue_manager.force_start_task(task)
else:
self.download_queue_manager.queue_task(task)
self.enable_download_list_update()
elif status == download.DownloadTask.CANCELLED:
# Cancelling a download allowed when downloading/queued
if task.status in (task.QUEUED, task.DOWNLOADING):
task.status = status
# Cancelling paused/failed downloads requires a call to .run()
elif task.status in (task.PAUSED, task.FAILED):
task.status = status
# Call run, so the partial file gets deleted
task.run()
elif status == download.DownloadTask.PAUSED:
# Pausing a download only when queued/downloading
if task.status in (task.DOWNLOADING, task.QUEUED):
task.status = status
elif status is None:
# Remove the selected task - cancel downloading/queued tasks
if task.status in (task.QUEUED, task.DOWNLOADING):
task.status = task.CANCELLED
model.remove(model.get_iter(row_reference.get_path()))
# Remember the URL, so we can tell the UI to update
try:
# We don't "see" this task anymore - remove it;
# this is needed, so update_episode_list_icons()
# below gets the correct list of "seen" tasks
self.download_tasks_seen.remove(task)
except KeyError as key_error:
pass
episode_urls.add(task.url)
# Tell the task that it has been removed (so it can clean up)
task.removed_from_list()
else:
# We can (hopefully) simply set the task status here
task.status = status
# Tell the podcasts tab to update icons for our removed podcasts
self.update_episode_list_icons(episode_urls)
# Update the tab title and downloads list
self.update_downloads_list()
def treeview_downloads_show_context_menu(self, treeview, event=None):
model, paths = self.treeview_handle_context_menu_click(treeview, event)
if not paths:
if not hasattr(treeview, 'is_rubber_banding_active'):
return True
else:
return not treeview.is_rubber_banding_active()
if event is None or event.button == 3:
selected_tasks, can_queue, can_cancel, can_pause, can_remove, can_force = \
self.downloads_list_get_selection(model, paths)
def make_menu_item(label, icon_name, tasks=None, status=None, sensitive=True, force_start=False, action=None):
# This creates a menu item for selection-wide actions
item = Gtk.ImageMenuItem.new_with_mnemonic(label)
if icon_name is not None:
item.set_image(Gtk.Image.new_from_icon_name(icon_name, Gtk.IconSize.MENU))
if action is not None:
item.connect('activate', action)
else:
item.connect('activate', lambda item: self._for_each_task_set_status(tasks, status, force_start))
item.set_sensitive(sensitive)
return item
def move_selected_items_up(menu_item):
selection = self.treeDownloads.get_selection()
model, selected_paths = selection.get_selected_rows()
for path in selected_paths:
index_above = path[0]-1
if index_above < 0:
return
task = model.get_value(
model.get_iter(path),
DownloadStatusModel.C_TASK)
model.move_before(
model.get_iter(path),
model.get_iter((index_above,)))
def move_selected_items_down(menu_item):
selection = self.treeDownloads.get_selection()
model, selected_paths = selection.get_selected_rows()
for path in reversed(selected_paths):
index_below = path[0]+1
if index_below >= len(model):
return
task = model.get_value(
model.get_iter(path),
DownloadStatusModel.C_TASK)
model.move_after(
model.get_iter(path),
model.get_iter((index_below,)))
menu = Gtk.Menu()
if can_force:
menu.append(make_menu_item(_('Start download now'), 'document-save', selected_tasks, download.DownloadTask.QUEUED, force_start=True))
else:
menu.append(make_menu_item(_('Download'), 'document-save', selected_tasks, download.DownloadTask.QUEUED, can_queue))
menu.append(make_menu_item(_('Cancel'), 'media-playback-stop', selected_tasks, download.DownloadTask.CANCELLED, can_cancel))
menu.append(make_menu_item(_('Pause'), 'media-playback-pause', selected_tasks, download.DownloadTask.PAUSED, can_pause))
menu.append(Gtk.SeparatorMenuItem())
menu.append(make_menu_item(_('Move up'), 'go-up', action=move_selected_items_up))
menu.append(make_menu_item(_('Move down'), 'go-down', action=move_selected_items_down))
menu.append(Gtk.SeparatorMenuItem())
menu.append(make_menu_item(_('Remove from list'), 'list-remove', selected_tasks, sensitive=can_remove))
menu.attach_to_widget(treeview)
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, event.button, event.time)
return True
def on_mark_episodes_as_old(self, item):
assert self.active_channel is not None
for episode in self.active_channel.get_all_episodes():
if not episode.was_downloaded(and_exists=True):
episode.mark(is_played=True)
self.update_podcast_list_model(selected=True)
self.update_episode_list_icons(all=True)
def on_open_download_folder(self, item):
assert self.active_channel is not None
util.gui_open(self.active_channel.save_dir)
def treeview_channels_show_context_menu(self, treeview, event=None):
model, paths = self.treeview_handle_context_menu_click(treeview, event)
if not paths:
return True
# Check for valid channel id, if there's no id then
# assume that it is a proxy channel or equivalent
# and cannot be operated with right click
if self.active_channel.id is None:
return True
if event is None or event.button == 3:
menu = Gtk.Menu()
item = Gtk.ImageMenuItem( _('Update podcast'))
item.set_image(Gtk.Image.new_from_icon_name('view-refresh', Gtk.IconSize.MENU))
item.set_action_name('win.updateChannel')
menu.append(item)
menu.append(Gtk.SeparatorMenuItem())
item = Gtk.MenuItem(_('Open download folder'))
item.connect('activate', self.on_open_download_folder)
menu.append(item)
menu.append(Gtk.SeparatorMenuItem())
item = Gtk.MenuItem(_('Mark episodes as old'))
item.connect('activate', self.on_mark_episodes_as_old)
menu.append(item)
item = Gtk.CheckMenuItem(_('Archive'))
item.set_active(self.active_channel.auto_archive_episodes)
item.connect('activate', self.on_channel_toggle_lock_activate)
menu.append(item)
item = Gtk.ImageMenuItem(_('Delete podcast'))
item.set_image(Gtk.Image.new_from_icon_name('edit-delete', Gtk.IconSize.MENU))
item.connect( 'activate', self.on_itemRemoveChannel_activate)
menu.append( item)
result = gpodder.user_extensions.on_channel_context_menu(self.active_channel)
if result:
menu.append(Gtk.SeparatorMenuItem())
for label, callback in result:
item = Gtk.MenuItem(label)
item.connect('activate', lambda item, callback: callback(self.active_channel), callback)
menu.append(item)
menu.append(Gtk.SeparatorMenuItem())
item = Gtk.ImageMenuItem(_('Podcast settings'))
item.set_image(Gtk.Image.new_from_icon_name('document-properties', Gtk.IconSize.MENU))
item.set_action_name('win.editChannel')
menu.append(item)
menu.attach_to_widget(treeview)
menu.show_all()
# Disable tooltips while we are showing the menu, so
# the tooltip will not appear over the menu
self.treeview_allow_tooltips(self.treeChannels, False)
menu.connect('deactivate', lambda menushell: self.treeview_allow_tooltips(self.treeChannels, True))
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, event.button, event.time)
return True
def cover_download_finished(self, channel, pixbuf):
"""
The Cover Downloader calls this when it has finished
downloading (or registering, if already downloaded)
a new channel cover, which is ready for displaying.
"""
util.idle_add(self.podcast_list_model.add_cover_by_channel,
channel, pixbuf)
@staticmethod
def build_filename(filename, extension):
filename = util.sanitize_filename(filename)
if not filename.endswith(extension):
filename += extension
return filename
def save_episodes_as_file(self, episodes):
PRIVATE_FOLDER_ATTRIBUTE = '_save_episodes_as_file_folder'
folder = getattr(self, PRIVATE_FOLDER_ATTRIBUTE, None)
(notCancelled, folder) = self.show_folder_select_dialog(initial_directory=folder)
setattr(self, PRIVATE_FOLDER_ATTRIBUTE, folder)
if notCancelled:
for episode in episodes:
if episode.was_downloaded(and_exists=True):
copy_from = episode.local_filename(create=False)
assert copy_from is not None
base, extension = os.path.splitext(copy_from)
filename = self.build_filename(episode.sync_filename(), extension)
copy_to = os.path.join(folder, filename)
try:
shutil.copyfile(copy_from, copy_to)
except (OSError, IOError) as e:
# Remove characters not supported by VFAT (#282)
new_filename = re.sub(r"[\"*/:<>?\\|]", "_", filename)
destination = os.path.join(folder, new_filename)
if (copy_to != destination):
shutil.copyfile(copy_from, destination)
else:
raise
def copy_episodes_bluetooth(self, episodes):
episodes_to_copy = [e for e in episodes if e.was_downloaded(and_exists=True)]
def convert_and_send_thread(episode):
for episode in episodes:
filename = episode.local_filename(create=False)
assert filename is not None
(base, ext) = os.path.splitext(filename)
destfile = self.build_fileName(episode.sync_filename(), ext)
destfile = os.path.join(tempfile.gettempdir(), destfile)
try:
shutil.copyfile(filename, destfile)
util.bluetooth_send_file(destfile)
except:
logger.error('Cannot copy "%s" to "%s".', filename, destfile)
self.notification(_('Error converting file.'), _('Bluetooth file transfer'), important=True)
util.delete_file(destfile)
util.run_in_background(lambda: convert_and_send_thread(episodes_to_copy))
def _add_sub_menu(self, menu, label):
root_item = Gtk.MenuItem(label)
menu.append(root_item)
sub_menu = Gtk.Menu()
root_item.set_submenu(sub_menu)
return sub_menu
def _submenu_item_activate_hack(self, item, callback, *args):
# See http://stackoverflow.com/questions/5221326/submenu-item-does-not-call-function-with-working-solution
# Note that we can't just call the callback on button-press-event, as
# it might be blocking (see http://gpodder.org/bug/1778), so we run
# this in the GUI thread at a later point in time (util.idle_add).
# Also, we also have to connect to the activate signal, as this is the
# only signal that is fired when keyboard navigation is used.
# It can happen that both (button-release-event and activate) signals
# are fired, and we must avoid calling the callback twice. We do this
# using a semaphore and only acquiring (but never releasing) it, making
# sure that the util.idle_add() call below is only ever called once.
only_once = threading.Semaphore(1)
def handle_event(item, event=None):
if only_once.acquire(False):
util.idle_add(callback, *args)
item.connect('button-press-event', handle_event)
item.connect('activate', handle_event)
def treeview_available_show_context_menu(self, treeview, event=None):
model, paths = self.treeview_handle_context_menu_click(treeview, event)
if not paths:
if not hasattr(treeview, 'is_rubber_banding_active'):
return True
else:
return not treeview.is_rubber_banding_active()
if event is None or event.button == 3:
episodes = self.get_selected_episodes()
any_locked = any(e.archive for e in episodes)
any_new = any(e.is_new for e in episodes)
downloaded = all(e.was_downloaded(and_exists=True) for e in episodes)
downloading = any(e.downloading for e in episodes)
menu = Gtk.Menu()
(can_play, can_download, can_cancel, can_delete, open_instead_of_play) = self.play_or_download()
if open_instead_of_play:
item = Gtk.ImageMenuItem(Gtk.STOCK_OPEN)
elif downloaded:
item = Gtk.ImageMenuItem(_('Play'))
item.set_image(Gtk.Image.new_from_icon_name('media-playback-start', Gtk.IconSize.MENU))
else:
if downloading:
item = Gtk.ImageMenuItem(_('Preview'))
else:
item = Gtk.ImageMenuItem(_('Stream'))
item.set_image(Gtk.Image.new_from_icon_name('media-playback-start', Gtk.IconSize.MENU))
item.set_sensitive(can_play)
item.connect('activate', self.on_playback_selected_episodes)
menu.append(item)
if not can_cancel:
item = Gtk.ImageMenuItem(_('Download'))
item.set_image(Gtk.Image.new_from_icon_name('document-save', Gtk.IconSize.MENU))
item.set_action_name('win.download')
menu.append(item)
else:
item = Gtk.ImageMenuItem.new_with_mnemonic(_('_Cancel'))
item.set_action_name('win.cancel')
menu.append(item)
item = Gtk.ImageMenuItem.new_with_mnemonic(_('_Delete'))
item.set_image(Gtk.Image.new_from_icon_name('edit-delete', Gtk.IconSize.MENU))
item.set_action_name('win.delete')
menu.append(item)
result = gpodder.user_extensions.on_episodes_context_menu(episodes)
if result:
menu.append(Gtk.SeparatorMenuItem())
submenus = {}
for label, callback in result:
key, sep, title = label.rpartition('/')
item = Gtk.ImageMenuItem(title)
self._submenu_item_activate_hack(item, callback, episodes)
if key:
if key not in submenus:
sub_menu = self._add_sub_menu(menu, key)
submenus[key] = sub_menu
else:
sub_menu = submenus[key]
sub_menu.append(item)
else:
menu.append(item)
# Ok, this probably makes sense to only display for downloaded files
if downloaded:
menu.append(Gtk.SeparatorMenuItem())
share_menu = self._add_sub_menu(menu, _('Send to'))
item = Gtk.ImageMenuItem(_('Local folder'))
item.set_image(Gtk.Image.new_from_icon_name('folder', Gtk.IconSize.MENU))
self._submenu_item_activate_hack(item, self.save_episodes_as_file, episodes)
share_menu.append(item)
if self.bluetooth_available:
item = Gtk.ImageMenuItem(_('Bluetooth device'))
item.set_image(Gtk.Image.new_from_icon_name('bluetooth', Gtk.IconSize.MENU))
self._submenu_item_activate_hack(item, self.copy_episodes_bluetooth, episodes)
share_menu.append(item)
menu.append(Gtk.SeparatorMenuItem())
item = Gtk.CheckMenuItem(_('New'))
item.set_active(any_new)
if any_new:
item.connect('activate', lambda w: self.mark_selected_episodes_old())
else:
item.connect('activate', lambda w: self.mark_selected_episodes_new())
menu.append(item)
if downloaded:
item = Gtk.CheckMenuItem(_('Archive'))
item.set_active(any_locked)
item.connect('activate', lambda w: self.on_item_toggle_lock_activate( w, False, not any_locked))
menu.append(item)
menu.append(Gtk.SeparatorMenuItem())
# Single item, add episode information menu item
item = Gtk.ImageMenuItem(_('Episode details'))
item.set_image(Gtk.Image.new_from_icon_name( 'dialog-information', Gtk.IconSize.MENU))
item.set_action_name('win.toggleShownotes')
menu.append(item)
menu.attach_to_widget(treeview)
menu.show_all()
# Disable tooltips while we are showing the menu, so
# the tooltip will not appear over the menu
self.treeview_allow_tooltips(self.treeAvailable, False)
menu.connect('deactivate', lambda menushell: self.treeview_allow_tooltips(self.treeAvailable, True))
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, event.button, event.time)
return True
def set_title(self, new_title):
self.default_title = new_title
self.gPodder.set_title(new_title)
def update_episode_list_icons(self, urls=None, selected=False, all=False):
"""
Updates the status icons in the episode list.
If urls is given, it should be a list of URLs
of episodes that should be updated.
If urls is None, set ONE OF selected, all to
True (the former updates just the selected
episodes and the latter updates all episodes).
"""
descriptions = self.config.episode_list_descriptions
if urls is not None:
# We have a list of URLs to walk through
self.episode_list_model.update_by_urls(urls, descriptions)
elif selected and not all:
# We should update all selected episodes
selection = self.treeAvailable.get_selection()
model, paths = selection.get_selected_rows()
for path in reversed(paths):
iter = model.get_iter(path)
self.episode_list_model.update_by_filter_iter(iter, descriptions)
elif all and not selected:
# We update all (even the filter-hidden) episodes
self.episode_list_model.update_all(descriptions)
else:
# Wrong/invalid call - have to specify at least one parameter
raise ValueError('Invalid call to update_episode_list_icons')
def episode_list_status_changed(self, episodes):
self.update_episode_list_icons(set(e.url for e in episodes))
self.update_podcast_list_model(set(e.channel.url for e in episodes))
self.db.commit()
def streaming_possible(self):
# User has to have a media player set on the Desktop, or else we
# would probably open the browser when giving a URL to xdg-open..
return (self.config.player and self.config.player != 'default')
def playback_episodes_for_real(self, episodes):
groups = collections.defaultdict(list)
for episode in episodes:
file_type = episode.file_type()
if file_type == 'video' and self.config.videoplayer and \
self.config.videoplayer != 'default':
player = self.config.videoplayer
elif file_type == 'audio' and self.config.player and \
self.config.player != 'default':
player = self.config.player
else:
player = 'default'
# Mark episode as played in the database
episode.playback_mark()
self.mygpo_client.on_playback([episode])
fmt_ids = youtube.get_fmt_ids(self.config.youtube)
vimeo_fmt = self.config.vimeo.fileformat
allow_partial = (player != 'default')
filename = episode.get_playback_url(fmt_ids, vimeo_fmt, allow_partial)
# Determine the playback resume position - if the file
# was played 100%, we simply start from the beginning
resume_position = episode.current_position
if resume_position == episode.total_time:
resume_position = 0
# If Panucci is configured, use D-Bus to call it
if player == 'panucci':
try:
PANUCCI_NAME = 'org.panucci.panucciInterface'
PANUCCI_PATH = '/panucciInterface'
PANUCCI_INTF = 'org.panucci.panucciInterface'
o = gpodder.dbus_session_bus.get_object(PANUCCI_NAME, PANUCCI_PATH)
i = dbus.Interface(o, PANUCCI_INTF)
def on_reply(*args):
pass
def error_handler(filename, err):
logger.error('Exception in D-Bus call: %s', str(err))
# Fallback: use the command line client
for command in util.format_desktop_command('panucci', \
[filename]):
logger.info('Executing: %s', repr(command))
subprocess.Popen(command)
on_error = lambda err: error_handler(filename, err)
# This method only exists in Panucci > 0.9 ('new Panucci')
i.playback_from(filename, resume_position, \
reply_handler=on_reply, error_handler=on_error)
continue # This file was handled by the D-Bus call
except Exception as e:
logger.error('Calling Panucci using D-Bus', exc_info=True)
groups[player].append(filename)
# Open episodes with system default player
if 'default' in groups:
for filename in groups['default']:
logger.debug('Opening with system default: %s', filename)
util.gui_open(filename)
del groups['default']
# For each type now, go and create play commands
for group in groups:
for command in util.format_desktop_command(group, groups[group], resume_position):
logger.debug('Executing: %s', repr(command))
subprocess.Popen(command)
# Persist episode status changes to the database
self.db.commit()
# Flush updated episode status
if self.mygpo_client.can_access_webservice():
self.mygpo_client.flush()
def playback_episodes(self, episodes):
# We need to create a list, because we run through it more than once
episodes = list(Model.sort_episodes_by_pubdate(e for e in episodes if \
e.was_downloaded(and_exists=True) or self.streaming_possible()))
try:
self.playback_episodes_for_real(episodes)
except Exception as e:
logger.error('Error in playback!', exc_info=True)
self.show_message(_('Please check your media player settings in the preferences dialog.'), \
_('Error opening player'))
self.episode_list_status_changed(episodes)
def play_or_download(self):
if self.wNotebook.get_current_page() > 0:
self.toolCancel.set_sensitive(True)
return (False, False, False, False, False, False)
( can_play, can_download, can_cancel, can_delete ) = (False,)*4
( is_played, is_locked ) = (False,)*2
open_instead_of_play = False
selection = self.treeAvailable.get_selection()
if selection.count_selected_rows() > 0:
(model, paths) = selection.get_selected_rows()
for path in paths:
try:
episode = model.get_value(model.get_iter(path), EpisodeListModel.C_EPISODE)
if episode is None:
logger.info('Invalid episode at path %s', str(path))
continue
except TypeError as te:
logger.error('Invalid episode at path %s', str(path))
continue
if episode.file_type() not in ('audio', 'video'):
open_instead_of_play = True
if episode.was_downloaded():
can_play = episode.was_downloaded(and_exists=True)
is_played = not episode.is_new
is_locked = episode.archive
if not can_play:
can_download = True
else:
if episode.downloading:
can_cancel = True
else:
can_download = True
can_download = can_download and not can_cancel
can_play = self.streaming_possible() or (can_play and not can_cancel and not can_download)
can_delete = not can_cancel
if open_instead_of_play:
self.toolPlay.set_stock_id(Gtk.STOCK_OPEN)
else:
self.toolPlay.set_stock_id(Gtk.STOCK_MEDIA_PLAY)
self.toolPlay.set_sensitive(can_play)
self.toolDownload.set_sensitive(can_download)
self.toolCancel.set_sensitive(can_cancel)
self.cancel_action.set_enabled(can_cancel)
self.download_action.set_enabled(can_download)
self.open_action.set_enabled(can_play and open_instead_of_play)
self.play_action.set_enabled(can_play and not open_instead_of_play)
self.delete_action.set_enabled(can_delete)
self.toggle_episode_new_action.set_enabled(can_play)
self.toggle_episode_lock_action.set_enabled(can_play)
# XXX: how to hide menu items?
#self.itemOpenSelected.set_visible(open_instead_of_play)
#self.itemPlaySelected.set_visible(not open_instead_of_play)
return (can_play, can_download, can_cancel, can_delete, open_instead_of_play)
def on_cbMaxDownloads_toggled(self, widget, *args):
self.spinMaxDownloads.set_sensitive(self.cbMaxDownloads.get_active())
def on_cbLimitDownloads_toggled(self, widget, *args):
self.spinLimitDownloads.set_sensitive(self.cbLimitDownloads.get_active())
def episode_new_status_changed(self, urls):
self.update_podcast_list_model()
self.update_episode_list_icons(urls)
def update_podcast_list_model(self, urls=None, selected=False, select_url=None,
sections_changed=False):
"""Update the podcast list treeview model
If urls is given, it should list the URLs of each
podcast that has to be updated in the list.
If selected is True, only update the model contents
for the currently-selected podcast - nothing more.
The caller can optionally specify "select_url",
which is the URL of the podcast that is to be
selected in the list after the update is complete.
This only works if the podcast list has to be
reloaded; i.e. something has been added or removed
since the last update of the podcast list).
"""
selection = self.treeChannels.get_selection()
model, iter = selection.get_selected()
is_section = lambda r: r[PodcastListModel.C_URL] == '-'
is_separator = lambda r: r[PodcastListModel.C_SEPARATOR]
sections_active = any(is_section(x) for x in self.podcast_list_model)
if self.config.podcast_list_view_all:
# Update "all episodes" view in any case (if enabled)
self.podcast_list_model.update_first_row()
# List model length minus 1, because of "All"
list_model_length = len(self.podcast_list_model) - 1
else:
list_model_length = len(self.podcast_list_model)
force_update = (sections_active != self.config.podcast_list_sections or
sections_changed)
# Filter items in the list model that are not podcasts, so we get the
# correct podcast list count (ignore section headers and separators)
is_not_podcast = lambda r: is_section(r) or is_separator(r)
list_model_length -= len(list(filter(is_not_podcast, self.podcast_list_model)))
if selected and not force_update:
# very cheap! only update selected channel
if iter is not None:
# If we have selected the "all episodes" view, we have
# to update all channels for selected episodes:
if self.config.podcast_list_view_all and \
self.podcast_list_model.iter_is_first_row(iter):
urls = self.get_podcast_urls_from_selected_episodes()
self.podcast_list_model.update_by_urls(urls)
else:
# Otherwise just update the selected row (a podcast)
self.podcast_list_model.update_by_filter_iter(iter)
if self.config.podcast_list_sections:
self.podcast_list_model.update_sections()
elif list_model_length == len(self.channels) and not force_update:
# we can keep the model, but have to update some
if urls is None:
# still cheaper than reloading the whole list
self.podcast_list_model.update_all()
else:
# ok, we got a bunch of urls to update
self.podcast_list_model.update_by_urls(urls)
if self.config.podcast_list_sections:
self.podcast_list_model.update_sections()
else:
if model and iter and select_url is None:
# Get the URL of the currently-selected podcast
select_url = model.get_value(iter, PodcastListModel.C_URL)
# Update the podcast list model with new channels
self.podcast_list_model.set_channels(self.db, self.config, self.channels)
try:
selected_iter = model.get_iter_first()
# Find the previously-selected URL in the new
# model if we have an URL (else select first)
if select_url is not None:
pos = model.get_iter_first()
while pos is not None:
url = model.get_value(pos, PodcastListModel.C_URL)
if url == select_url:
selected_iter = pos
break
pos = model.iter_next(pos)
if selected_iter is not None:
selection.select_iter(selected_iter)
self.on_treeChannels_cursor_changed(self.treeChannels)
except:
logger.error('Cannot select podcast in list', exc_info=True)
def on_episode_list_filter_changed(self, has_episodes):
self.play_or_download()
def update_episode_list_model(self):
if self.channels and self.active_channel is not None:
self.treeAvailable.get_selection().unselect_all()
self.treeAvailable.scroll_to_point(0, 0)
descriptions = self.config.episode_list_descriptions
self.episode_list_model.replace_from_channel(self.active_channel, descriptions)
else:
self.episode_list_model.clear()
@dbus.service.method(gpodder.dbus_interface)
def offer_new_episodes(self, channels=None):
new_episodes = self.get_new_episodes(channels)
if new_episodes:
self.new_episodes_show(new_episodes)
return True
return False
def add_podcast_list(self, podcasts, auth_tokens=None):
"""Subscribe to a list of podcast given (title, url) pairs
If auth_tokens is given, it should be a dictionary
mapping URLs to (username, password) tuples."""
if auth_tokens is None:
auth_tokens = {}
existing_urls = set(podcast.url for podcast in self.channels)
# For a given URL, the desired title (or None)
title_for_url = {}
# Sort and split the URL list into five buckets
queued, failed, existing, worked, authreq = [], [], [], [], []
for input_title, input_url in podcasts:
url = util.normalize_feed_url(input_url)
# Check if it's a YouTube feed, and if we have an API key, auto-resolve the channel
url = youtube.resolve_v3_url(url, self.config.youtube.api_key_v3)
if url is None:
# Fail this one because the URL is not valid
failed.append(input_url)
elif url in existing_urls:
# A podcast already exists in the list for this URL
existing.append(url)
# XXX: Should we try to update the title of the existing
# subscription from input_title here if it is different?
else:
# This URL has survived the first round - queue for add
title_for_url[url] = input_title
queued.append(url)
if url != input_url and input_url in auth_tokens:
auth_tokens[url] = auth_tokens[input_url]
error_messages = {}
redirections = {}
progress = ProgressIndicator(_('Adding podcasts'), \
_('Please wait while episode information is downloaded.'), \
parent=self.get_dialog_parent())
def on_after_update():
progress.on_finished()
# Report already-existing subscriptions to the user
if existing:
title = _('Existing subscriptions skipped')
message = _('You are already subscribed to these podcasts:') \
+ '\n\n' + '\n'.join(cgi.escape(url) for url in existing)
self.show_message(message, title, widget=self.treeChannels)
# Report subscriptions that require authentication
retry_podcasts = {}
if authreq:
for url in authreq:
title = _('Podcast requires authentication')
message = _('Please login to %s:') % (cgi.escape(url),)
success, auth_tokens = self.show_login_dialog(title, message)
if success:
retry_podcasts[url] = auth_tokens
else:
# Stop asking the user for more login data
retry_podcasts = {}
for url in authreq:
error_messages[url] = _('Authentication failed')
failed.append(url)
break
# Report website redirections
for url in redirections:
title = _('Website redirection detected')
message = _('The URL %(url)s redirects to %(target)s.') \
+ '\n\n' + _('Do you want to visit the website now?')
message = message % {'url': url, 'target': redirections[url]}
if self.show_confirmation(message, title):
util.open_website(url)
else:
break
# Report failed subscriptions to the user
if failed:
title = _('Could not add some podcasts')
message = _('Some podcasts could not be added to your list:') \
+ '\n\n' + '\n'.join(cgi.escape('%s: %s' % (url, \
error_messages.get(url, _('Unknown')))) for url in failed)
self.show_message(message, title, important=True)
# Upload subscription changes to gpodder.net
self.mygpo_client.on_subscribe(worked)
# Fix URLs if mygpo has rewritten them
self.rewrite_urls_mygpo()
# If only one podcast was added, select it after the update
if len(worked) == 1:
url = worked[0]
else:
url = None
# Update the list of subscribed podcasts
self.update_podcast_list_model(select_url=url)
# If we have authentication data to retry, do so here
if retry_podcasts:
podcasts = [(title_for_url.get(url), url)
for url in list(retry_podcasts.keys())]
self.add_podcast_list(podcasts, retry_podcasts)
# This will NOT show new episodes for podcasts that have
# been added ("worked"), but it will prevent problems with
# multiple dialogs being open at the same time ;)
return
# Offer to download new episodes
episodes = []
for podcast in self.channels:
if podcast.url in worked:
episodes.extend(podcast.get_all_episodes())
if episodes:
episodes = list(Model.sort_episodes_by_pubdate(episodes, \
reverse=True))
self.new_episodes_show(episodes, \
selected=[e.check_is_new() for e in episodes])
@util.run_in_background
def thread_proc():
# After the initial sorting and splitting, try all queued podcasts
length = len(queued)
for index, url in enumerate(queued):
title = title_for_url.get(url)
progress.on_progress(float(index)/float(length))
progress.on_message(title or url)
try:
# The URL is valid and does not exist already - subscribe!
channel = self.model.load_podcast(url=url, create=True, \
authentication_tokens=auth_tokens.get(url, None), \
max_episodes=self.config.max_episodes_per_feed)
try:
username, password = util.username_password_from_url(url)
except ValueError as ve:
username, password = (None, None)
if title is not None:
# Prefer title from subscription source (bug 1711)
channel.title = title
if username is not None and channel.auth_username is None and \
password is not None and channel.auth_password is None:
channel.auth_username = username
channel.auth_password = password
channel.save()
self._update_cover(channel)
except feedcore.AuthenticationRequired:
if url in auth_tokens:
# Fail for wrong authentication data
error_messages[url] = _('Authentication failed')
failed.append(url)
else:
# Queue for login dialog later
authreq.append(url)
continue
except feedcore.WifiLogin as error:
redirections[url] = error.data
failed.append(url)
error_messages[url] = _('Redirection detected')
continue
except Exception as e:
logger.error('Subscription error: %s', e, exc_info=True)
error_messages[url] = str(e)
failed.append(url)
continue
assert channel is not None
worked.append(channel.url)
util.idle_add(on_after_update)
def find_episode(self, podcast_url, episode_url):
"""Find an episode given its podcast and episode URL
The function will return a PodcastEpisode object if
the episode is found, or None if it's not found.
"""
for podcast in self.channels:
if podcast_url == podcast.url:
for episode in podcast.get_all_episodes():
if episode_url == episode.url:
return episode
return None
def process_received_episode_actions(self):
"""Process/merge episode actions from gpodder.net
This function will merge all changes received from
the server to the local database and update the
status of the affected episodes as necessary.
"""
indicator = ProgressIndicator(_('Merging episode actions'), \
_('Episode actions from gpodder.net are merged.'), \
False, self.get_dialog_parent())
Gtk.main_iteration()
self.mygpo_client.process_episode_actions(self.find_episode)
indicator.on_finished()
self.db.commit()
def _update_cover(self, channel):
if channel is not None:
self.cover_downloader.request_cover(channel)
def show_update_feeds_buttons(self):
# Make sure that the buttons for updating feeds
# appear - this should happen after a feed update
self.hboxUpdateFeeds.hide()
self.btnUpdateFeeds.show()
self.update_action.set_enabled(True)
self.update_channel_action.set_enabled(True)
def on_btnCancelFeedUpdate_clicked(self, widget):
if not self.feed_cache_update_cancelled:
self.pbFeedUpdate.set_text(_('Cancelling...'))
self.feed_cache_update_cancelled = True
self.btnCancelFeedUpdate.set_sensitive(False)
else:
self.show_update_feeds_buttons()
def update_feed_cache(self, channels=None,
show_new_episodes_dialog=True):
if not util.connection_available():
self.show_message(_('Please connect to a network, then try again.'),
_('No network connection'), important=True)
return
# Fix URLs if mygpo has rewritten them
self.rewrite_urls_mygpo()
if channels is None:
# Only update podcasts for which updates are enabled
channels = [c for c in self.channels if not c.pause_subscription]
self.update_action.set_enabled(False)
self.update_channel_action.set_enabled(False)
self.feed_cache_update_cancelled = False
self.btnCancelFeedUpdate.show()
self.btnCancelFeedUpdate.set_sensitive(True)
self.btnCancelFeedUpdate.set_image(Gtk.Image.new_from_icon_name('process-stop', Gtk.IconSize.BUTTON))
self.hboxUpdateFeeds.show_all()
self.btnUpdateFeeds.hide()
count = len(channels)
text = N_('Updating %(count)d feed...', 'Updating %(count)d feeds...', count) % {'count':count}
self.pbFeedUpdate.set_text(text)
self.pbFeedUpdate.set_fraction(0)
@util.run_in_background
def update_feed_cache_proc():
updated_channels = []
for updated, channel in enumerate(channels):
if self.feed_cache_update_cancelled:
break
def indicate_updating_podcast(channel):
d = {'podcast': channel.title, 'position': updated+1, 'total': count}
progression = _('Updating %(podcast)s (%(position)d/%(total)d)') % d
self.pbFeedUpdate.set_text(progression)
try:
util.idle_add(indicate_updating_podcast, channel)
channel.update(max_episodes=self.config.max_episodes_per_feed)
self._update_cover(channel)
except Exception as e:
d = {'url': cgi.escape(channel.url), 'message': cgi.escape(str(e))}
if d['message']:
message = _('Error while updating %(url)s: %(message)s')
else:
message = _('The feed at %(url)s could not be updated.')
self.notification(message % d, _('Error while updating feed'), widget=self.treeChannels)
logger.error('Error: %s', str(e), exc_info=True)
updated_channels.append(channel)
def update_progress(channel):
self.update_podcast_list_model([channel.url])
# If the currently-viewed podcast is updated, reload episodes
if self.active_channel is not None and \
self.active_channel == channel:
logger.debug('Updated channel is active, updating UI')
self.update_episode_list_model()
self.pbFeedUpdate.set_fraction(float(updated+1)/float(count))
util.idle_add(update_progress, channel)
def update_feed_cache_finish_callback():
# Process received episode actions for all updated URLs
self.process_received_episode_actions()
# If we are currently viewing "All episodes", update its episode list now
if self.active_channel is not None and \
getattr(self.active_channel, 'ALL_EPISODES_PROXY', False):
self.update_episode_list_model()
if self.feed_cache_update_cancelled:
# The user decided to abort the feed update
self.show_update_feeds_buttons()
# Only search for new episodes in podcasts that have been
# updated, not in other podcasts (for single-feed updates)
episodes = self.get_new_episodes([c for c in updated_channels])
if self.config.downloads.chronological_order:
# download older episodes first
episodes = list(Model.sort_episodes_by_pubdate(episodes))
if not episodes:
# Nothing new here - but inform the user
self.pbFeedUpdate.set_fraction(1.0)
self.pbFeedUpdate.set_text(_('No new episodes'))
self.feed_cache_update_cancelled = True
self.btnCancelFeedUpdate.show()
self.btnCancelFeedUpdate.set_sensitive(True)
self.update_action.set_enabled(True)
self.btnCancelFeedUpdate.set_image(Gtk.Image.new_from_icon_name('edit-clear', Gtk.IconSize.BUTTON))
else:
count = len(episodes)
# New episodes are available
self.pbFeedUpdate.set_fraction(1.0)
if self.config.auto_download == 'download':
self.download_episode_list(episodes)
title = N_('Downloading %(count)d new episode.', 'Downloading %(count)d new episodes.', count) % {'count':count}
self.show_message(title, _('New episodes available'))
elif self.config.auto_download == 'queue':
self.download_episode_list_paused(episodes)
title = N_('%(count)d new episode added to download list.', '%(count)d new episodes added to download list.', count) % {'count':count}
self.show_message(title, _('New episodes available'))
else:
if (show_new_episodes_dialog and
self.config.auto_download == 'show'):
self.new_episodes_show(episodes, notification=True)
else: # !show_new_episodes_dialog or auto_download == 'ignore'
message = N_('%(count)d new episode available', '%(count)d new episodes available', count) % {'count':count}
self.pbFeedUpdate.set_text(message)
self.show_update_feeds_buttons()
util.idle_add(update_feed_cache_finish_callback)
def on_gPodder_delete_event(self, *args):
"""Called when the GUI wants to close the window
Displays a confirmation dialog (and closes/hides gPodder)
"""
if self.confirm_quit():
self.close_gpodder()
return True
def confirm_quit(self):
"""Called when the GUI wants to close the window
Displays a confirmation dialog
"""
downloading = self.download_status_model.are_downloads_in_progress()
if downloading:
dialog = Gtk.MessageDialog(self.gPodder, Gtk.DialogFlags.MODAL, Gtk.MessageType.QUESTION, Gtk.ButtonsType.NONE)
dialog.add_button(Gtk.STOCK_CANCEL, Gtk.ResponseType.CANCEL)
quit_button = dialog.add_button(Gtk.STOCK_QUIT, Gtk.ResponseType.CLOSE)
title = _('Quit gPodder')
message = _('You are downloading episodes. You can resume downloads the next time you start gPodder. Do you want to quit now?')
dialog.set_title(title)
dialog.set_markup('%s\n\n%s'%(title, message))
quit_button.grab_focus()
result = dialog.run()
dialog.destroy()
return result == Gtk.ResponseType.CLOSE
else:
return True
def close_gpodder(self):
""" clean everything and exit properly
"""
# Cancel any running background updates of the episode list model
self.episode_list_model.background_update = None
self.gPodder.hide()
# Notify all tasks to to carry out any clean-up actions
self.download_status_model.tell_all_tasks_to_quit()
while Gtk.events_pending():
Gtk.main_iteration()
self.core.shutdown()
self.application.remove_window(self.gPodder)
def delete_episode_list(self, episodes, confirm=True, skip_locked=True,
callback=None):
if not episodes:
return False
if skip_locked:
episodes = [e for e in episodes if not e.archive]
if not episodes:
title = _('Episodes are locked')
message = _('The selected episodes are locked. Please unlock the episodes that you want to delete before trying to delete them.')
self.notification(message, title, widget=self.treeAvailable)
return False
count = len(episodes)
title = N_('Delete %(count)d episode?', 'Delete %(count)d episodes?', count) % {'count':count}
message = _('Deleting episodes removes downloaded files.')
if confirm and not self.show_confirmation(message, title):
return False
progress = ProgressIndicator(_('Deleting episodes'), \
_('Please wait while episodes are deleted'), \
parent=self.get_dialog_parent())
def finish_deletion(episode_urls, channel_urls):
progress.on_finished()
# Episodes have been deleted - persist the database
self.db.commit()
self.update_episode_list_icons(episode_urls)
self.update_podcast_list_model(channel_urls)
self.play_or_download()
@util.run_in_background
def thread_proc():
episode_urls = set()
channel_urls = set()
episodes_status_update = []
for idx, episode in enumerate(episodes):
progress.on_progress(idx/len(episodes))
if not episode.archive or not skip_locked:
progress.on_message(episode.title)
episode.delete_from_disk()
episode_urls.add(episode.url)
channel_urls.add(episode.channel.url)
episodes_status_update.append(episode)
# Notify the web service about the status update + upload
if self.mygpo_client.can_access_webservice():
self.mygpo_client.on_delete(episodes_status_update)
self.mygpo_client.flush()
if callback is None:
util.idle_add(finish_deletion, episode_urls, channel_urls)
else:
util.idle_add(callback, episode_urls, channel_urls, progress)
return True
def on_itemRemoveOldEpisodes_activate(self, action, param):
self.show_delete_episodes_window()
def show_delete_episodes_window(self, channel=None):
"""Offer deletion of episodes
If channel is None, offer deletion of all episodes.
Otherwise only offer deletion of episodes in the channel.
"""
columns = (
('markup_delete_episodes', None, None, _('Episode')),
)
msg_older_than = N_('Select older than %(count)d day', 'Select older than %(count)d days', self.config.episode_old_age)
selection_buttons = {
_('Select played'): lambda episode: not episode.is_new,
_('Select finished'): lambda episode: episode.is_finished(),
msg_older_than % {'count':self.config.episode_old_age}: lambda episode: episode.age_in_days() > self.config.episode_old_age,
}
instructions = _('Select the episodes you want to delete:')
if channel is None:
channels = self.channels
else:
channels = [channel]
episodes = []
for channel in channels:
for episode in channel.get_episodes(gpodder.STATE_DOWNLOADED):
# Disallow deletion of locked episodes that still exist
if not episode.archive or not episode.file_exists():
episodes.append(episode)
selected = [not e.is_new or not e.file_exists() for e in episodes]
gPodderEpisodeSelector(self.main_window, title = _('Delete episodes'), instructions = instructions, \
episodes = episodes, selected = selected, columns = columns, \
stock_ok_button = 'edit-delete', callback = self.delete_episode_list, \
selection_buttons = selection_buttons, _config=self.config)
def on_selected_episodes_status_changed(self):
# The order of the updates here is important! When "All episodes" is
# selected, the update of the podcast list model depends on the episode
# list selection to determine which podcasts are affected. Updating
# the episode list could remove the selection if a filter is active.
self.update_podcast_list_model(selected=True)
self.update_episode_list_icons(selected=True)
self.db.commit()
def mark_selected_episodes_new(self):
for episode in self.get_selected_episodes():
episode.mark_new()
self.on_selected_episodes_status_changed()
def mark_selected_episodes_old(self):
for episode in self.get_selected_episodes():
episode.mark_old()
self.on_selected_episodes_status_changed()
def on_item_toggle_played_activate(self, action, param):
for episode in self.get_selected_episodes():
episode.mark(is_played=episode.is_new)
self.on_selected_episodes_status_changed()
def on_item_toggle_lock_activate(self, widget, toggle=True, new_value=False):
for episode in self.get_selected_episodes():
if toggle:
episode.mark(is_locked=not episode.archive)
else:
episode.mark(is_locked=new_value)
self.on_selected_episodes_status_changed()
def on_channel_toggle_lock_activate(self, widget, toggle=True, new_value=False):
if self.active_channel is None:
return
self.active_channel.auto_archive_episodes = not self.active_channel.auto_archive_episodes
self.active_channel.save()
for episode in self.active_channel.get_all_episodes():
episode.mark(is_locked=self.active_channel.auto_archive_episodes)
self.update_podcast_list_model(selected=True)
self.update_episode_list_icons(all=True)
def on_itemUpdateChannel_activate(self, *params):
if self.active_channel is None:
title = _('No podcast selected')
message = _('Please select a podcast in the podcasts list to update.')
self.show_message( message, title, widget=self.treeChannels)
return
# Dirty hack to check for "All episodes" (see gpodder.gtkui.model)
if getattr(self.active_channel, 'ALL_EPISODES_PROXY', False):
self.update_feed_cache()
else:
self.update_feed_cache(channels=[self.active_channel])
def on_itemUpdate_activate(self, action=None, param=None):
# Check if we have outstanding subscribe/unsubscribe actions
self.on_add_remove_podcasts_mygpo()
if self.channels:
self.update_feed_cache()
else:
def show_welcome_window():
def on_show_example_podcasts(widget):
welcome_window.main_window.response(Gtk.ResponseType.CANCEL)
self.on_itemImportChannels_activate(None)
def on_add_podcast_via_url(widget):
welcome_window.main_window.response(Gtk.ResponseType.CANCEL)
self.on_itemAddChannel_activate(None)
def on_setup_my_gpodder(widget):
welcome_window.main_window.response(Gtk.ResponseType.CANCEL)
self.on_download_subscriptions_from_mygpo(None)
welcome_window = gPodderWelcome(self.main_window,
center_on_widget=self.main_window,
on_show_example_podcasts=on_show_example_podcasts,
on_add_podcast_via_url=on_add_podcast_via_url,
on_setup_my_gpodder=on_setup_my_gpodder)
welcome_window.main_window.run()
welcome_window.main_window.destroy()
util.idle_add(show_welcome_window)
def download_episode_list_paused(self, episodes):
self.download_episode_list(episodes, True)
def download_episode_list(self, episodes, add_paused=False, force_start=False):
enable_update = False
if self.config.downloads.chronological_order:
# Download episodes in chronological order (older episodes first)
episodes = list(Model.sort_episodes_by_pubdate(episodes))
for episode in episodes:
logger.debug('Downloading episode: %s', episode.title)
if not episode.was_downloaded(and_exists=True):
task_exists = False
for task in self.download_tasks_seen:
if episode.url == task.url:
task_exists = True
if task.status not in (task.DOWNLOADING, task.QUEUED):
if force_start:
self.download_queue_manager.force_start_task(task)
else:
self.download_queue_manager.queue_task(task)
enable_update = True
continue
if task_exists:
continue
try:
task = download.DownloadTask(episode, self.config)
except Exception as e:
d = {'episode': episode.title, 'message': str(e)}
message = _('Download error while downloading %(episode)s: %(message)s')
self.show_message(message % d, _('Download error'), important=True)
logger.error('While downloading %s', episode.title, exc_info=True)
continue
# New Task, we must wait on the GTK Loop
self.download_status_model.register_task(task)
def queue_task(task):
if add_paused:
task.status = task.PAUSED
else:
self.mygpo_client.on_download([task.episode])
if force_start:
self.download_queue_manager.force_start_task(task)
else:
self.download_queue_manager.queue_task(task)
# Executes after task has been registered
util.idle_add(queue_task, task)
enable_update = True
if enable_update:
self.enable_download_list_update()
# Flush updated episode status
if self.mygpo_client.can_access_webservice():
self.mygpo_client.flush()
def cancel_task_list(self, tasks):
if not tasks:
return
for task in tasks:
if task.status in (task.QUEUED, task.DOWNLOADING):
task.status = task.CANCELLED
elif task.status == task.PAUSED:
task.status = task.CANCELLED
# Call run, so the partial file gets deleted
task.run()
self.update_episode_list_icons([task.url for task in tasks])
self.play_or_download()
# Update the tab title and downloads list
self.update_downloads_list()
def new_episodes_show(self, episodes, notification=False, selected=None):
columns = (
('markup_new_episodes', None, None, _('Episode')),
)
instructions = _('Select the episodes you want to download:')
if self.new_episodes_window is not None:
self.new_episodes_window.main_window.destroy()
self.new_episodes_window = None
def download_episodes_callback(episodes):
self.new_episodes_window = None
self.download_episode_list(episodes)
if selected is None:
# Select all by default
selected = [True]*len(episodes)
self.new_episodes_window = gPodderEpisodeSelector(self.main_window, \
title=_('New episodes available'), \
instructions=instructions, \
episodes=episodes, \
columns=columns, \
selected=selected, \
stock_ok_button = 'gpodder-download', \
callback=download_episodes_callback, \
remove_callback=lambda e: e.mark_old(), \
remove_action=_('Mark as old'), \
remove_finished=self.episode_new_status_changed, \
_config=self.config, \
show_notification=False)
def on_itemDownloadAllNew_activate(self, action, param):
if not self.offer_new_episodes():
self.show_message(_('Please check for new episodes later.'), \
_('No new episodes available'))
def get_new_episodes(self, channels=None):
return [e for c in channels or self.channels for e in
[e for e in c.get_all_episodes() if e.check_is_new()]]
def commit_changes_to_database(self):
"""This will be called after the sync process is finished"""
self.db.commit()
def on_itemShowToolbar_activate(self, action, param):
state = action.get_state()
self.config.show_toolbar = not state
action.set_state(GLib.Variant.new_boolean(not state))
def on_itemShowDescription_activate(self, action, param):
state = action.get_state()
self.config.episode_list_descriptions = not state
action.set_state(GLib.Variant.new_boolean(not state))
def on_item_view_hide_boring_podcasts_toggled(self, action, param):
state = action.get_state()
self.config.podcast_list_hide_boring = not state
action.set_state(GLib.Variant.new_boolean(not state))
self.apply_podcast_list_hide_boring()
def on_item_view_episodes_changed(self, action, param):
self.config.episode_list_view_mode = getattr(EpisodeListModel, param.get_string()) or EpisodeListModel.VIEW_ALL
action.set_state(param)
self.episode_list_model.set_view_mode(self.config.episode_list_view_mode)
self.apply_podcast_list_hide_boring()
def apply_podcast_list_hide_boring(self):
if self.config.podcast_list_hide_boring:
self.podcast_list_model.set_view_mode(self.config.episode_list_view_mode)
else:
self.podcast_list_model.set_view_mode(-1)
def on_download_subscriptions_from_mygpo(self, action=None):
def after_login():
title = _('Subscriptions on %(server)s') \
% {'server': self.config.mygpo.server}
dir = gPodderPodcastDirectory(self.gPodder,
_config=self.config,
custom_title=title,
add_podcast_list=self.add_podcast_list,
hide_url_entry=True)
url = self.mygpo_client.get_download_user_subscriptions_url()
dir.download_opml_file(url)
title = _('Login to gpodder.net')
message = _('Please login to download your subscriptions.')
def on_register_button_clicked():
util.open_website('http://gpodder.net/register/')
success, (root_url, username, password) = self.show_login_dialog(title, message,
self.config.mygpo.server,
self.config.mygpo.username, self.config.mygpo.password,
register_callback=on_register_button_clicked,
ask_server=True)
if not success:
return
self.config.mygpo.server = root_url
self.config.mygpo.username = username
self.config.mygpo.password = password
util.idle_add(after_login)
def on_itemAddChannel_activate(self, action=None, param=None):
self._add_podcast_dialog = gPodderAddPodcast(self.gPodder, \
add_podcast_list=self.add_podcast_list)
def on_itemEditChannel_activate(self, action, param=None):
if self.active_channel is None:
title = _('No podcast selected')
message = _('Please select a podcast in the podcasts list to edit.')
self.show_message( message, title, widget=self.treeChannels)
return
gPodderChannel(self.main_window,
channel=self.active_channel,
update_podcast_list_model=self.update_podcast_list_model,
cover_downloader=self.cover_downloader,
sections=set(c.section for c in self.channels),
clear_cover_cache=self.podcast_list_model.clear_cover_cache,
_config=self.config)
def on_itemMassUnsubscribe_activate(self, action, param):
columns = (
('title', None, None, _('Podcast')),
)
# We're abusing the Episode Selector for selecting Podcasts here,
# but it works and looks good, so why not? -- thp
gPodderEpisodeSelector(self.main_window, \
title=_('Delete podcasts'), \
instructions=_('Select the podcast you want to delete.'), \
episodes=self.channels, \
columns=columns, \
size_attribute=None, \
stock_ok_button=_('Delete'), \
callback=self.remove_podcast_list, \
_config=self.config)
def remove_podcast_list(self, channels, confirm=True):
if not channels:
return
if len(channels) == 1:
title = _('Deleting podcast')
info = _('Please wait while the podcast is deleted')
message = _('This podcast and all its episodes will be PERMANENTLY DELETED.\r\nAre you sure you want to continue?')
else:
title = _('Deleting podcasts')
info = _('Please wait while the podcasts are deleted')
message = _('These podcasts and all their episodes will be PERMANENTLY DELETED.\r\nAre you sure you want to continue?')
if confirm and not self.show_confirmation(message, title):
return
progress = ProgressIndicator(title, info, parent=self.get_dialog_parent())
def finish_deletion(select_url):
# Upload subscription list changes to the web service
self.mygpo_client.on_unsubscribe([c.url for c in channels])
# Re-load the channels and select the desired new channel
self.update_podcast_list_model(select_url=select_url)
progress.on_finished()
@util.run_in_background
def thread_proc():
select_url = None
for idx, channel in enumerate(channels):
# Update the UI for correct status messages
progress.on_progress(idx/len(channels))
progress.on_message(channel.title)
# Delete downloaded episodes
channel.remove_downloaded()
# cancel any active downloads from this channel
for episode in channel.get_all_episodes():
if episode.downloading:
episode.download_task.cancel()
if len(channels) == 1:
# get the URL of the podcast we want to select next
if channel in self.channels:
position = self.channels.index(channel)
else:
position = -1
if position == len(self.channels)-1:
# this is the last podcast, so select the URL
# of the item before this one (i.e. the "new last")
select_url = self.channels[position-1].url
else:
# there is a podcast after the deleted one, so
# we simply select the one that comes after it
select_url = self.channels[position+1].url
# Remove the channel and clean the database entries
channel.delete()
# Clean up downloads and download directories
common.clean_up_downloads()
# The remaining stuff is to be done in the GTK main thread
util.idle_add(finish_deletion, select_url)
def on_itemRemoveChannel_activate(self, widget, *args):
if self.active_channel is None:
title = _('No podcast selected')
message = _('Please select a podcast in the podcasts list to remove.')
self.show_message( message, title, widget=self.treeChannels)
return
self.remove_podcast_list([self.active_channel])
def get_opml_filter(self):
filter = Gtk.FileFilter()
filter.add_pattern('*.opml')
filter.add_pattern('*.xml')
filter.set_name(_('OPML files')+' (*.opml, *.xml)')
return filter
def on_item_import_from_file_activate(self, action, filename=None):
if filename is None:
dlg = Gtk.FileChooserDialog(title=_('Import from OPML'),
parent=self.main_window,
action=Gtk.FileChooserAction.OPEN)
dlg.add_button(Gtk.STOCK_CANCEL, Gtk.ResponseType.CANCEL)
dlg.add_button(Gtk.STOCK_OPEN, Gtk.ResponseType.OK)
dlg.set_filter(self.get_opml_filter())
response = dlg.run()
filename = None
if response == Gtk.ResponseType.OK:
filename = dlg.get_filename()
dlg.destroy()
if filename is not None:
dir = gPodderPodcastDirectory(self.gPodder, _config=self.config, \
custom_title=_('Import podcasts from OPML file'), \
add_podcast_list=self.add_podcast_list,
hide_url_entry=True)
dir.download_opml_file(filename)
def on_itemExportChannels_activate(self, widget, *args):
if not self.channels:
title = _('Nothing to export')
message = _('Your list of podcast subscriptions is empty. Please subscribe to some podcasts first before trying to export your subscription list.')
self.show_message(message, title, widget=self.treeChannels)
return
dlg = Gtk.FileChooserDialog(title=_('Export to OPML'), parent=self.gPodder, action=Gtk.FileChooserAction.SAVE)
dlg.add_button(Gtk.STOCK_CANCEL, Gtk.ResponseType.CANCEL)
dlg.add_button(Gtk.STOCK_SAVE, Gtk.ResponseType.OK)
dlg.set_filter(self.get_opml_filter())
response = dlg.run()
if response == Gtk.ResponseType.OK:
filename = dlg.get_filename()
dlg.destroy()
exporter = opml.Exporter( filename)
if filename is not None and exporter.write(self.channels):
count = len(self.channels)
title = N_('%(count)d subscription exported', '%(count)d subscriptions exported', count) % {'count':count}
self.show_message(_('Your podcast list has been successfully exported.'), title, widget=self.treeChannels)
else:
self.show_message( _('Could not export OPML to file. Please check your permissions.'), _('OPML export failed'), important=True)
else:
dlg.destroy()
def on_itemImportChannels_activate(self, widget, *args):
self._podcast_directory = gPodderPodcastDirectory(self.main_window,
_config=self.config,
add_podcast_list=self.add_podcast_list)
def on_homepage_activate(self, widget, *args):
util.open_website(gpodder.__url__)
def check_for_updates(self, silent):
"""Check for updates and (optionally) show a message
If silent=False, a message will be shown even if no updates are
available (set silent=False when the check is manually triggered).
"""
try:
up_to_date, version, released, days = util.get_update_info()
except Exception as e:
if silent:
logger.warn('Could not check for updates.', exc_info=True)
else:
title = _('Could not check for updates')
message = _('Please try again later.')
self.show_message(message, title, important=True)
return
if up_to_date and not silent:
title = _('No updates available')
message = _('You have the latest version of gPodder.')
self.show_message(message, title, important=True)
if not up_to_date:
title = _('New version available')
message = '\n'.join([
_('Installed version: %s') % gpodder.__version__,
_('Newest version: %s') % version,
_('Release date: %s') % released,
'',
_('Download the latest version from gpodder.org?'),
])
if self.show_confirmation(message, title):
util.open_website('http://gpodder.org/downloads')
def on_wNotebook_switch_page(self, notebook, page, page_num):
if page_num == 0:
self.play_or_download()
# The message area in the downloads tab should be hidden
# when the user switches away from the downloads tab
if self.message_area is not None:
self.message_area.hide()
self.message_area = None
else:
self.toolDownload.set_sensitive(False)
self.toolPlay.set_sensitive(False)
self.toolCancel.set_sensitive(False)
def on_treeChannels_row_activated(self, widget, path, *args):
# double-click action of the podcast list or enter
self.treeChannels.set_cursor(path)
def on_treeChannels_cursor_changed(self, widget, *args):
( model, iter ) = self.treeChannels.get_selection().get_selected()
if model is not None and iter is not None:
old_active_channel = self.active_channel
self.active_channel = model.get_value(iter, PodcastListModel.C_CHANNEL)
if self.active_channel == old_active_channel:
return
# Dirty hack to check for "All episodes" (see gpodder.gtkui.model)
if getattr(self.active_channel, 'ALL_EPISODES_PROXY', False):
self.edit_channel_action.set_enabled(False)
else:
self.edit_channel_action.set_enabled(True)
else:
self.active_channel = None
self.edit_channel_action.set_enabled(False)
self.update_episode_list_model()
def on_btnEditChannel_clicked(self, widget, *args):
self.on_itemEditChannel_activate( widget, args)
def get_podcast_urls_from_selected_episodes(self):
"""Get a set of podcast URLs based on the selected episodes"""
return set(episode.channel.url for episode in \
self.get_selected_episodes())
def get_selected_episodes(self):
"""Get a list of selected episodes from treeAvailable"""
selection = self.treeAvailable.get_selection()
model, paths = selection.get_selected_rows()
episodes = [model.get_value(model.get_iter(path), EpisodeListModel.C_EPISODE) for path in paths]
episodes = [e for e in episodes if e is not None]
return episodes
def on_playback_selected_episodes(self, *params):
self.playback_episodes(self.get_selected_episodes())
def on_shownotes_selected_episodes(self, *params):
episodes = self.get_selected_episodes()
self.shownotes_object.toggle_pane_visibility(episodes)
def on_download_selected_episodes(self, action_or_widget, param=None):
episodes = self.get_selected_episodes()
self.download_episode_list(episodes)
self.update_episode_list_icons([episode.url for episode in episodes])
self.play_or_download()
def on_treeAvailable_row_activated(self, widget, path, view_column):
"""Double-click/enter action handler for treeAvailable"""
self.on_shownotes_selected_episodes(widget)
def restart_auto_update_timer(self):
if self._auto_update_timer_source_id is not None:
logger.debug('Removing existing auto update timer.')
GObject.source_remove(self._auto_update_timer_source_id)
self._auto_update_timer_source_id = None
if self.config.auto_update_feeds and \
self.config.auto_update_frequency:
interval = 60*1000*self.config.auto_update_frequency
logger.debug('Setting up auto update timer with interval %d.',
self.config.auto_update_frequency)
self._auto_update_timer_source_id = GObject.timeout_add(\
interval, self._on_auto_update_timer)
def _on_auto_update_timer(self):
if not util.connection_available():
logger.debug('Skipping auto update (no connection available)')
return True
logger.debug('Auto update timer fired.')
self.update_feed_cache()
# Ask web service for sub changes (if enabled)
if self.mygpo_client.can_access_webservice():
self.mygpo_client.flush()
return True
def on_treeDownloads_row_activated(self, widget, *args):
# Use the standard way of working on the treeview
selection = self.treeDownloads.get_selection()
(model, paths) = selection.get_selected_rows()
selected_tasks = [(Gtk.TreeRowReference.new(model, path), model.get_value(model.get_iter(path), 0)) for path in paths]
for tree_row_reference, task in selected_tasks:
if task.status in (task.DOWNLOADING, task.QUEUED):
task.status = task.PAUSED
elif task.status in (task.CANCELLED, task.PAUSED, task.FAILED):
self.download_queue_manager.queue_task(task)
self.enable_download_list_update()
elif task.status == task.DONE:
model.remove(model.get_iter(tree_row_reference.get_path()))
self.play_or_download()
# Update the tab title and downloads list
self.update_downloads_list()
def on_item_cancel_download_activate(self, *params):
if self.wNotebook.get_current_page() == 0:
selection = self.treeAvailable.get_selection()
(model, paths) = selection.get_selected_rows()
urls = [model.get_value(model.get_iter(path), \
self.episode_list_model.C_URL) for path in paths]
selected_tasks = [task for task in self.download_tasks_seen \
if task.url in urls]
else:
selection = self.treeDownloads.get_selection()
(model, paths) = selection.get_selected_rows()
selected_tasks = [model.get_value(model.get_iter(path), \
self.download_status_model.C_TASK) for path in paths]
self.cancel_task_list(selected_tasks)
def on_btnCancelAll_clicked(self, widget, *args):
self.cancel_task_list(self.download_tasks_seen)
def on_btnDownloadedDelete_clicked(self, widget, *args):
episodes = self.get_selected_episodes()
if len(episodes) == 1:
self.delete_episode_list(episodes, skip_locked=False)
else:
self.delete_episode_list(episodes)
def on_key_press(self, widget, event):
# Allow tab switching with Ctrl + PgUp/PgDown/Tab
if event.get_state() & Gdk.ModifierType.CONTROL_MASK:
current_page = self.wNotebook.get_current_page()
if event.keyval in (Gdk.KEY_Page_Up, Gdk.KEY_ISO_Left_Tab):
if current_page == 0:
current_page = self.wNotebook.get_n_pages()
self.wNotebook.set_current_page(current_page - 1)
return True
elif event.keyval in (Gdk.KEY_Page_Down, Gdk.KEY_Tab):
if current_page == self.wNotebook.get_n_pages() - 1:
current_page = -1
self.wNotebook.set_current_page(current_page + 1)
return True
elif event.keyval == Gdk.KEY_Delete:
if isinstance(widget.get_focus(), Gtk.Entry):
logger.debug("Entry has focus, ignoring Delete")
else:
self.main_window.activate_action('delete')
return True
return False
def uniconify_main_window(self):
if self.is_iconified():
# We need to hide and then show the window in WMs like Metacity
# or KWin4 to move the window to the active workspace
# (see http://gpodder.org/bug/1125)
self.gPodder.hide()
self.gPodder.show()
self.gPodder.present()
def iconify_main_window(self):
if not self.is_iconified():
self.gPodder.iconify()
@dbus.service.method(gpodder.dbus_interface)
def show_gui_window(self):
parent = self.get_dialog_parent()
parent.present()
@dbus.service.method(gpodder.dbus_interface)
def subscribe_to_url(self, url):
# Strip leading application protocol, so these URLs work:
# gpodder://example.com/episodes.rss
# gpodder:https://example.org/podcast.xml
if url.startswith('gpodder:'):
url = url[len('gpodder:'):]
while url.startswith('/'):
url = url[1:]
self._add_podcast_dialog = gPodderAddPodcast(self.gPodder,
add_podcast_list=self.add_podcast_list,
preset_url=url)
@dbus.service.method(gpodder.dbus_interface)
def mark_episode_played(self, filename):
if filename is None:
return False
for channel in self.channels:
for episode in channel.get_all_episodes():
fn = episode.local_filename(create=False, check_only=True)
if fn == filename:
episode.mark(is_played=True)
self.db.commit()
self.update_episode_list_icons([episode.url])
self.update_podcast_list_model([episode.channel.url])
return True
return False
def extensions_podcast_update_cb(self, podcast):
logger.debug('extensions_podcast_update_cb(%s)', podcast)
self.update_feed_cache(channels=[podcast],
show_new_episodes_dialog=False)
def extensions_episode_download_cb(self, episode):
logger.debug('extension_episode_download_cb(%s)', episode)
self.download_episode_list(episodes=[episode])
def on_sync_to_device_activate(self, widget, episodes=None, force_played=True):
self.sync_ui = gPodderSyncUI(self.config, self.notification,
self.main_window,
self.show_confirmation,
self.application.on_itemPreferences_activate,
self.channels,
self.download_status_model,
self.download_queue_manager,
self.enable_download_list_update,
self.commit_changes_to_database,
self.delete_episode_list)
self.sync_ui.on_synchronize_episodes(self.channels, episodes, force_played)
def on_update_youtube_subscriptions_activate(self, action, param):
if not self.config.youtube.api_key_v3:
if self.show_confirmation('\n'.join((_('Please register a YouTube API key and set it in the preferences.'),
_('Would you like to set up an API key now?'))), _('API key required')):
self.application.on_itemPreferences_activate(self, None)
return
failed_urls = []
migrated_users = []
for podcast in self.channels:
url, user = youtube.for_each_feed_pattern(lambda url, channel: (url, channel), podcast.url, (None, None))
if url is not None and user is not None:
try:
logger.info('Getting channels for YouTube user %s (%s)', user, url)
new_urls = youtube.get_channels_for_user(user, self.config.youtube.api_key_v3)
logger.debug('YouTube channels retrieved: %r', new_urls)
if len(new_urls) == 0 and youtube.get_youtube_id(url) is not None:
logger.info('No need to update %s', url)
continue
if len(new_urls) != 1:
failed_urls.append((url, _('No unique URL found')))
continue
new_url = new_urls[0]
if new_url in set(x.url for x in self.model.get_podcasts()):
failed_urls.append((url, _('Already subscribed')))
continue
logger.info('New feed location: %s => %s', url, new_url)
podcast.url = new_url
podcast.save()
migrated_users.append(user)
except Exception as e:
logger.error('Exception happened while updating download list.', exc_info=True)
self.show_message(_('Make sure the API key is correct. Error: %(message)s') % {'message': str(e)},
_('Error getting YouTube channels'), important=True)
if migrated_users:
self.show_message('\n'.join(migrated_users), _('Successfully migrated subscriptions'))
elif not failed_urls:
self.show_message(_('Subscriptions are up to date'))
if failed_urls:
self.show_message('\n'.join([_('These URLs failed:'), ''] + ['{0}: {1}'.format(url, message)
for url, message in failed_urls]),
_('Could not migrate some subscriptions'), important=True)
def on_extension_enabled(self, extension):
if getattr(extension, 'on_ui_object_available', None) is not None:
extension.on_ui_object_available('gpodder-gtk', self)
if getattr(extension, 'on_ui_initialized', None) is not None:
extension.on_ui_initialized(self.model,
self.extensions_podcast_update_cb,
self.extensions_episode_download_cb)
self.inject_extensions_menu()
def on_extension_disabled(self, extension):
self.inject_extensions_menu()
class gPodderApplication(Gtk.Application):
def __init__(self, options):
Gtk.Application.__init__(self, application_id='org.gpodder.gpodder',
flags=Gio.ApplicationFlags.FLAGS_NONE)
self.window = None
self.options = options;
self.connect('window-removed', self.on_window_removed)
def create_actions(self):
action = Gio.SimpleAction.new('about', None)
action.connect('activate', self.on_about)
self.add_action(action)
action = Gio.SimpleAction.new('quit', None)
action.connect('activate', self.on_quit)
self.add_action(action)
action = Gio.SimpleAction.new('help', None)
action.connect('activate', self.on_help_activate)
self.add_action(action)
action = Gio.SimpleAction.new('preferences', None)
action.connect('activate', self.on_itemPreferences_activate)
self.add_action(action)
action = Gio.SimpleAction.new('gotoMygpo', None)
action.connect('activate', self.on_goto_mygpo)
self.add_action(action)
action = Gio.SimpleAction.new('checkForUpdates', None)
action.connect('activate', self.on_check_for_updates_activate)
self.add_action(action)
def do_startup(self):
Gtk.Application.do_startup(self)
self.create_actions()
builder = Gtk.Builder()
builder.set_translation_domain(gpodder.textdomain)
for ui_folder in gpodder.ui_folders:
filename = os.path.join(ui_folder, 'gtk/menus.ui')
if os.path.exists(filename):
builder.add_from_file(filename)
break
menubar = builder.get_object('menubar')
if menubar is None:
logger.error('Cannot find gtk/menus.ui in %r, exiting' % gpodder.ui_folders)
sys.exit(1)
self.menu_view_columns = builder.get_object('menuViewColumns')
self.set_menubar(menubar)
self.set_app_menu(builder.get_object('app-menu'))
for i in range(EpisodeListModel.PROGRESS_STEPS + 1):
pixbuf = draw_cake_pixbuf(i /
EpisodeListModel.PROGRESS_STEPS)
icon_name = 'gpodder-progress-%d' % i
Gtk.IconTheme.add_builtin_icon(icon_name, pixbuf.get_width(), pixbuf)
Gtk.Window.set_default_icon_name('gpodder')
#Gtk.AboutDialog.set_url_hook(lambda dlg, link, data: util.open_website(link), None)
try:
dbus_main_loop = dbus.glib.DBusGMainLoop(set_as_default=True)
gpodder.dbus_session_bus = dbus.SessionBus(dbus_main_loop)
self.bus_name = dbus.service.BusName(gpodder.dbus_bus_name, bus=gpodder.dbus_session_bus)
except dbus.exceptions.DBusException as dbe:
logger.warn('Cannot get "on the bus".', exc_info=True)
dlg = Gtk.MessageDialog(None, Gtk.DialogFlags.MODAL, Gtk.MessageType.ERROR, \
Gtk.ButtonsType.CLOSE, _('Cannot start gPodder'))
dlg.format_secondary_markup(_('D-Bus error: %s') % (str(dbe),))
dlg.set_title('gPodder')
dlg.run()
dlg.destroy()
sys.exit(0)
def do_activate(self):
# We only allow a single window and raise any existing ones
if not self.window:
# Windows are associated with the application
# when the last one is closed the application shuts down
self.window = gPodder(self, self.bus_name, core.Core(UIConfig, model_class=Model), self.options)
if gpodder.ui.osx:
from gpodder.gtkui import macosx
# Handle "subscribe to podcast" events from firefox
macosx.register_handlers(self.window)
self.window.gPodder.present()
def on_about(self, action, param):
dlg = Gtk.Dialog(_('About gPodder'), self.window.gPodder, \
Gtk.DialogFlags.MODAL)
dlg.add_button(Gtk.STOCK_CLOSE, Gtk.ResponseType.OK).show()
dlg.set_resizable(False)
bg = Gtk.Box(orientation=Gtk.Orientation.HORIZONTAL, spacing=6)
pb = GdkPixbuf.Pixbuf.new_from_file_at_size(gpodder.icon_file, 160, 160)
bg.pack_start(Gtk.Image.new_from_pixbuf(pb), False, False, 0)
vb = Gtk.Box(orientation=Gtk.Orientation.VERTICAL, spacing=6)
label = Gtk.Label()
label.set_alignment(0, 0.5)
label.set_markup('\n'.join(x.strip() for x in """
gPodder {version} ({date})
{copyright}
License: {license}
Website ยท Bug Tracker
""".format(version=gpodder.__version__,
date=gpodder.__date__,
copyright=gpodder.__copyright__,
license=gpodder.__license__,
bugs_url='https://github.com/gpodder/gpodder/issues',
url=cgi.escape(gpodder.__url__)).strip().split('\n')))
vb.pack_start(label, False, False, 0)
bg.pack_start(vb, False, False, 0)
bg.pack_start(Gtk.Label(), False, False, 0)
dlg.vbox.pack_start(bg, False, False, 0)
dlg.connect('response', lambda dlg, response: dlg.destroy())
dlg.vbox.show_all()
dlg.run()
def on_quit(self, *args):
self.window.on_gPodder_delete_event()
def on_window_removed(self, *args):
self.quit()
def on_help_activate(self, action, param):
util.open_website('https://gpodder.github.io/docs/')
def on_itemPreferences_activate(self, action, param=None):
gPodderPreferences(self.window.gPodder, \
_config=self.window.config, \
user_apps_reader=self.window.user_apps_reader, \
parent_window=self.window.main_window, \
mygpo_client=self.window.mygpo_client, \
on_send_full_subscriptions=self.window.on_send_full_subscriptions, \
on_itemExportChannels_activate=self.window.on_itemExportChannels_activate, \
on_extension_enabled=self.on_extension_enabled,
on_extension_disabled=self.on_extension_disabled)
def on_goto_mygpo(self, action, param):
self.window.mygpo_client.open_website()
def on_check_for_updates_activate(self, action, param):
self.window.check_for_updates(silent=False)
def on_extension_enabled(self, extension):
self.window.on_extension_enabled(extension)
def on_extension_disabled(self, extension):
self.window.on_extension_disabled(extension)
def main(options=None):
GObject.set_application_name('gPodder')
gp = gPodderApplication(options)
gp.run()
sys.exit(0)