182 lines
6.2 KiB
Python
182 lines
6.2 KiB
Python
# vim: ft=python fileencoding=utf-8 sts=4 sw=4 et:
|
|
|
|
# Copyright 2014-2015 Florian Bruhin (The Compiler) <mail@qutebrowser.org>
|
|
#
|
|
# This file is part of qutebrowser.
|
|
#
|
|
# qutebrowser 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.
|
|
#
|
|
# qutebrowser 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 qutebrowser. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
"""The ListView to display downloads in."""
|
|
|
|
import functools
|
|
|
|
import sip
|
|
from PyQt5.QtCore import pyqtSlot, QSize, Qt, QTimer
|
|
from PyQt5.QtWidgets import QListView, QSizePolicy, QMenu
|
|
|
|
from qutebrowser.browser import downloads
|
|
from qutebrowser.config import style
|
|
from qutebrowser.utils import qtutils, utils, objreg
|
|
|
|
|
|
def update_geometry(obj):
|
|
"""Weird WORKAROUND for some weird PyQt bug (probably).
|
|
|
|
This actually should be a method of DownloadView, but for some reason the
|
|
rowsInserted/rowsRemoved signals don't get disconnected from this method
|
|
when the DownloadView is deleted from Qt (e.g. by closing a window).
|
|
|
|
Here we check if obj ("self") was deleted and just ignore the event if so.
|
|
|
|
Original bug: https://github.com/The-Compiler/qutebrowser/issues/167
|
|
Workaround bug: https://github.com/The-Compiler/qutebrowser/issues/171
|
|
"""
|
|
def _update_geometry():
|
|
"""Actually update the geometry if the object still exists."""
|
|
if sip.isdeleted(obj):
|
|
return
|
|
obj.updateGeometry()
|
|
|
|
# If we don't use a singleShot QTimer, the geometry isn't updated correctly
|
|
# and won't include the new item.
|
|
QTimer.singleShot(0, _update_geometry)
|
|
|
|
|
|
class DownloadView(QListView):
|
|
|
|
"""QListView which shows currently running downloads as a bar.
|
|
|
|
Attributes:
|
|
_menu: The QMenu which is currently displayed.
|
|
_model: The currently set model.
|
|
"""
|
|
|
|
STYLESHEET = """
|
|
QListView {
|
|
background-color: {{ color['downloads.bg.bar'] }};
|
|
font: {{ font['downloads'] }};
|
|
}
|
|
|
|
QListView::item {
|
|
padding-right: 2px;
|
|
}
|
|
"""
|
|
|
|
def __init__(self, win_id, parent=None):
|
|
super().__init__(parent)
|
|
style.set_register_stylesheet(self)
|
|
self.setResizeMode(QListView.Adjust)
|
|
self.setVerticalScrollBarPolicy(Qt.ScrollBarAlwaysOff)
|
|
self.setSizePolicy(QSizePolicy.MinimumExpanding, QSizePolicy.Fixed)
|
|
self.setFlow(QListView.LeftToRight)
|
|
self.setSpacing(1)
|
|
self._menu = None
|
|
model = objreg.get('download-manager', scope='window', window=win_id)
|
|
model.rowsInserted.connect(functools.partial(update_geometry, self))
|
|
model.rowsRemoved.connect(functools.partial(update_geometry, self))
|
|
model.dataChanged.connect(functools.partial(update_geometry, self))
|
|
self.setModel(model)
|
|
self.setWrapping(True)
|
|
self.setContextMenuPolicy(Qt.CustomContextMenu)
|
|
self.customContextMenuRequested.connect(self.show_context_menu)
|
|
self.clicked.connect(self.on_clicked)
|
|
|
|
def __repr__(self):
|
|
model = self.model()
|
|
if model is None:
|
|
count = 'None'
|
|
else:
|
|
count = model.rowCount()
|
|
return utils.get_repr(self, count=count)
|
|
|
|
@pyqtSlot('QModelIndex')
|
|
def on_clicked(self, index):
|
|
"""Handle clicking of an item.
|
|
|
|
Args:
|
|
index: The QModelIndex of the clicked item.
|
|
"""
|
|
if not index.isValid():
|
|
return
|
|
item = self.model().data(index, downloads.ModelRole.item)
|
|
if item.done and item.successful:
|
|
item.open_file()
|
|
self.model().remove_item(item)
|
|
|
|
def _get_menu_actions(self, item):
|
|
"""Get the available context menu actions for a given DownloadItem.
|
|
|
|
Args:
|
|
item: The DownloadItem to get the actions for, or None.
|
|
|
|
Return:
|
|
A list of either:
|
|
- (QAction, callable) tuples.
|
|
- (None, None) for a separator
|
|
"""
|
|
model = self.model()
|
|
actions = []
|
|
if item is None:
|
|
pass
|
|
elif item.done:
|
|
if item.successful:
|
|
actions.append(("Open", item.open_file))
|
|
else:
|
|
actions.append(("Retry", item.retry))
|
|
actions.append(("Remove",
|
|
functools.partial(model.remove_item, item)))
|
|
else:
|
|
actions.append(("Cancel", item.cancel))
|
|
if model.can_clear():
|
|
actions.append((None, None))
|
|
actions.append(("Remove all finished", model.download_clear))
|
|
return actions
|
|
|
|
@pyqtSlot('QPoint')
|
|
def show_context_menu(self, point):
|
|
"""Show the context menu."""
|
|
index = self.indexAt(point)
|
|
if index.isValid():
|
|
item = self.model().data(index, downloads.ModelRole.item)
|
|
else:
|
|
item = None
|
|
self._menu = QMenu(self)
|
|
actions = self._get_menu_actions(item)
|
|
for (name, handler) in actions:
|
|
if name is None and handler is None:
|
|
self._menu.addSeparator()
|
|
else:
|
|
action = self._menu.addAction(name)
|
|
action.triggered.connect(handler)
|
|
if actions:
|
|
self._menu.popup(self.viewport().mapToGlobal(point))
|
|
|
|
def minimumSizeHint(self):
|
|
"""Override minimumSizeHint so the size is correct in a layout."""
|
|
return self.sizeHint()
|
|
|
|
def sizeHint(self):
|
|
"""Return sizeHint based on the view contents."""
|
|
idx = self.model().last_index()
|
|
bottom = self.visualRect(idx).bottom()
|
|
if bottom != -1:
|
|
margins = self.contentsMargins()
|
|
height = (bottom + margins.top() + margins.bottom() +
|
|
2 * self.spacing())
|
|
size = QSize(0, height)
|
|
else:
|
|
size = QSize(0, 0)
|
|
qtutils.ensure_valid(size)
|
|
return size
|