qutebrowser/qutebrowser/browser/browsertab.py

717 lines
22 KiB
Python
Raw Normal View History

2016-06-13 11:50:58 +02:00
# vim: ft=python fileencoding=utf-8 sts=4 sw=4 et:
# Copyright 2016 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/>.
"""Base class for a wrapper over QWebView/QWebEngineView."""
2016-06-13 15:05:31 +02:00
import itertools
2016-08-17 19:20:14 +02:00
from PyQt5.QtCore import pyqtSignal, pyqtSlot, QUrl, QObject, QSizeF, QTimer
2016-06-13 14:44:41 +02:00
from PyQt5.QtGui import QIcon
2016-08-17 18:02:24 +02:00
from PyQt5.QtWidgets import QWidget
2016-06-13 11:50:58 +02:00
2016-07-08 10:20:53 +02:00
from qutebrowser.keyinput import modeman
2016-06-15 13:02:24 +02:00
from qutebrowser.config import config
from qutebrowser.utils import (utils, objreg, usertypes, message, log, qtutils,
2016-08-11 16:38:45 +02:00
urlutils)
2016-08-03 13:08:25 +02:00
from qutebrowser.misc import miscwidgets
from qutebrowser.browser import mouse, hints
2016-06-13 17:49:52 +02:00
2016-06-13 11:50:58 +02:00
2016-06-13 15:05:31 +02:00
tab_id_gen = itertools.count(0)
2016-07-08 10:20:53 +02:00
def create(win_id, parent=None):
"""Get a QtWebKit/QtWebEngine tab object.
Args:
win_id: The window ID where the tab will be shown.
parent: The Qt parent to set.
"""
# Importing modules here so we don't depend on QtWebEngine without the
# argument and to avoid circular imports.
mode_manager = modeman.instance(win_id)
if objreg.get('args').backend == 'webengine':
from qutebrowser.browser.webengine import webenginetab
2016-07-10 17:23:08 +02:00
tab_class = webenginetab.WebEngineTab
2016-07-08 10:20:53 +02:00
else:
from qutebrowser.browser.webkit import webkittab
2016-07-10 17:23:08 +02:00
tab_class = webkittab.WebKitTab
2016-07-08 10:20:53 +02:00
return tab_class(win_id=win_id, mode_manager=mode_manager, parent=parent)
2016-07-04 13:35:38 +02:00
class WebTabError(Exception):
"""Base class for various errors."""
2016-06-13 15:05:31 +02:00
2016-08-17 19:20:14 +02:00
class TabData:
2016-07-04 16:12:58 +02:00
"""A simple namespace with a fixed set of attributes.
Attributes:
keep_icon: Whether the (e.g. cloned) icon should not be cleared on page
load.
2016-07-04 17:49:25 +02:00
inspector: The QWebInspector used for this webview.
viewing_source: Set if we're currently showing a source view.
open_target: How the next clicked link should be opened.
hint_target: Override for open_target for hints.
2016-07-04 16:12:58 +02:00
"""
2016-08-17 19:20:14 +02:00
def __init__(self):
self.keep_icon = False
self.viewing_source = False
self.inspector = None
self.open_target = usertypes.ClickTarget.normal
self.hint_target = None
def combined_target(self):
if self.hint_target is not None:
return self.hint_target
else:
return self.open_target
2016-07-04 16:12:58 +02:00
class AbstractPrinting:
"""Attribute of AbstractTab for printing the page."""
def __init__(self):
self._widget = None
def check_pdf_support(self):
raise NotImplementedError
def check_printer_support(self):
raise NotImplementedError
def to_pdf(self, filename):
raise NotImplementedError
def to_printer(self, printer):
raise NotImplementedError
2016-07-04 11:23:46 +02:00
class AbstractSearch(QObject):
"""Attribute of AbstractTab for doing searches.
Attributes:
text: The last thing this view was searched for.
_flags: The flags of the last search (needs to be set by subclasses).
2016-07-08 09:15:28 +02:00
_widget: The underlying WebView widget.
2016-07-04 11:23:46 +02:00
"""
def __init__(self, parent=None):
super().__init__(parent)
self._widget = None
2016-07-04 11:23:46 +02:00
self.text = None
2016-07-12 16:47:57 +02:00
def search(self, text, *, ignore_case=False, reverse=False,
result_cb=None):
2016-07-04 11:23:46 +02:00
"""Find the given text on the page.
Args:
text: The text to search for.
ignore_case: Search case-insensitively. (True/False/'smart')
reverse: Reverse search direction.
result_cb: Called with a bool indicating whether a match was found.
2016-07-04 11:23:46 +02:00
"""
raise NotImplementedError
def clear(self):
"""Clear the current search."""
raise NotImplementedError
def prev_result(self, *, result_cb=None):
"""Go to the previous result of the current search.
Args:
result_cb: Called with a bool indicating whether a match was found.
"""
2016-07-04 11:23:46 +02:00
raise NotImplementedError
def next_result(self, *, result_cb=None):
"""Go to the next result of the current search.
Args:
result_cb: Called with a bool indicating whether a match was found.
"""
2016-07-04 11:23:46 +02:00
raise NotImplementedError
2016-06-15 13:02:24 +02:00
class AbstractZoom(QObject):
"""Attribute of AbstractTab for controlling zoom.
Attributes:
_neighborlist: A NeighborList with the zoom levels.
_default_zoom_changed: Whether the zoom was changed from the default.
"""
def __init__(self, win_id, parent=None):
super().__init__(parent)
self._widget = None
2016-06-15 13:02:24 +02:00
self._win_id = win_id
self._default_zoom_changed = False
self._init_neighborlist()
2016-07-07 16:10:35 +02:00
objreg.get('config').changed.connect(self._on_config_changed)
2016-06-15 13:02:24 +02:00
2016-07-07 18:03:37 +02:00
# # FIXME:qtwebengine is this needed?
2016-06-15 13:02:24 +02:00
# # For some reason, this signal doesn't get disconnected automatically
# # when the WebView is destroyed on older PyQt versions.
# # See https://github.com/The-Compiler/qutebrowser/issues/390
# self.destroyed.connect(functools.partial(
# cfg.changed.disconnect, self.init_neighborlist))
@pyqtSlot(str, str)
2016-07-07 16:10:35 +02:00
def _on_config_changed(self, section, option):
if section == 'ui' and option in ['zoom-levels', 'default-zoom']:
2016-06-15 13:02:24 +02:00
if not self._default_zoom_changed:
factor = float(config.get('ui', 'default-zoom')) / 100
self._set_factor_internal(factor)
self._default_zoom_changed = False
self._init_neighborlist()
def _init_neighborlist(self):
"""Initialize self._neighborlist."""
levels = config.get('ui', 'zoom-levels')
self._neighborlist = usertypes.NeighborList(
levels, mode=usertypes.NeighborList.Modes.edge)
self._neighborlist.fuzzyval = config.get('ui', 'default-zoom')
def offset(self, offset):
"""Increase/Decrease the zoom level by the given offset.
Args:
offset: The offset in the zoom level list.
Return:
The new zoom percentage.
"""
level = self._neighborlist.getitem(offset)
self.set_factor(float(level) / 100, fuzzyval=False)
return level
def set_factor(self, factor, *, fuzzyval=True):
"""Zoom to a given zoom factor.
Args:
factor: The zoom factor as float.
fuzzyval: Whether to set the NeighborLists fuzzyval.
"""
if fuzzyval:
self._neighborlist.fuzzyval = int(factor * 100)
if factor < 0:
raise ValueError("Can't zoom to factor {}!".format(factor))
self._default_zoom_changed = True
self._set_factor_internal(factor)
def factor(self):
raise NotImplementedError
def set_default(self):
default_zoom = config.get('ui', 'default-zoom')
self._set_factor_internal(float(default_zoom) / 100)
2016-06-15 13:02:24 +02:00
2016-06-14 18:25:07 +02:00
class AbstractCaret(QObject):
2016-06-14 18:08:46 +02:00
"""Attribute of AbstractTab for caret browsing."""
def __init__(self, win_id, tab, mode_manager, parent=None):
2016-06-14 18:25:07 +02:00
super().__init__(parent)
2016-07-04 13:35:38 +02:00
self._tab = tab
2016-06-14 18:08:46 +02:00
self._win_id = win_id
self._widget = None
2016-06-14 18:08:46 +02:00
self.selection_enabled = False
mode_manager.entered.connect(self._on_mode_entered)
mode_manager.left.connect(self._on_mode_left)
2016-06-14 18:08:46 +02:00
2016-07-07 16:10:35 +02:00
def _on_mode_entered(self, mode):
2016-06-14 18:08:46 +02:00
raise NotImplementedError
2016-07-07 16:10:35 +02:00
def _on_mode_left(self):
2016-06-14 18:08:46 +02:00
raise NotImplementedError
def move_to_next_line(self, count=1):
raise NotImplementedError
def move_to_prev_line(self, count=1):
raise NotImplementedError
def move_to_next_char(self, count=1):
raise NotImplementedError
def move_to_prev_char(self, count=1):
raise NotImplementedError
def move_to_end_of_word(self, count=1):
raise NotImplementedError
def move_to_next_word(self, count=1):
raise NotImplementedError
def move_to_prev_word(self, count=1):
raise NotImplementedError
def move_to_start_of_line(self):
raise NotImplementedError
def move_to_end_of_line(self):
raise NotImplementedError
def move_to_start_of_next_block(self, count=1):
raise NotImplementedError
def move_to_start_of_prev_block(self, count=1):
raise NotImplementedError
def move_to_end_of_next_block(self, count=1):
raise NotImplementedError
def move_to_end_of_prev_block(self, count=1):
raise NotImplementedError
def move_to_start_of_document(self):
raise NotImplementedError
def move_to_end_of_document(self):
raise NotImplementedError
def toggle_selection(self):
raise NotImplementedError
def drop_selection(self):
raise NotImplementedError
def has_selection(self):
raise NotImplementedError
def selection(self, html=False):
raise NotImplementedError
2016-07-05 20:11:42 +02:00
def follow_selected(self, *, tab=False):
2016-07-04 13:35:38 +02:00
raise NotImplementedError
2016-06-14 18:08:46 +02:00
2016-06-14 17:32:36 +02:00
class AbstractScroller(QObject):
"""Attribute of AbstractTab to manage scroll position."""
perc_changed = pyqtSignal(int, int)
def __init__(self, tab, parent=None):
2016-06-14 17:32:36 +02:00
super().__init__(parent)
self._tab = tab
self._widget = None
2016-06-14 17:32:36 +02:00
def _init_widget(self, widget):
self._widget = widget
2016-06-14 17:32:36 +02:00
def pos_px(self):
raise NotImplementedError
def pos_perc(self):
raise NotImplementedError
def to_perc(self, x=None, y=None):
raise NotImplementedError
def to_point(self, point):
raise NotImplementedError
def delta(self, x=0, y=0):
raise NotImplementedError
def delta_page(self, x=0, y=0):
raise NotImplementedError
def up(self, count=1):
raise NotImplementedError
def down(self, count=1):
raise NotImplementedError
def left(self, count=1):
raise NotImplementedError
def right(self, count=1):
raise NotImplementedError
def top(self):
raise NotImplementedError
def bottom(self):
raise NotImplementedError
def page_up(self, count=1):
raise NotImplementedError
def page_down(self, count=1):
raise NotImplementedError
def at_top(self):
raise NotImplementedError
def at_bottom(self):
raise NotImplementedError
2016-06-14 11:03:34 +02:00
class AbstractHistory:
"""The history attribute of a AbstractTab."""
def __init__(self, tab):
2016-06-15 14:23:24 +02:00
self._tab = tab
2016-07-04 12:50:15 +02:00
self._history = None
2016-06-14 11:03:34 +02:00
2016-07-04 12:48:29 +02:00
def __len__(self):
2016-07-04 12:50:15 +02:00
return len(self._history)
2016-07-04 12:48:29 +02:00
2016-06-14 13:53:35 +02:00
def __iter__(self):
2016-07-04 12:50:15 +02:00
return iter(self._history.items())
2016-06-14 13:53:35 +02:00
def current_idx(self):
raise NotImplementedError
2016-06-14 11:03:34 +02:00
def back(self):
raise NotImplementedError
def forward(self):
raise NotImplementedError
def can_go_back(self):
raise NotImplementedError
def can_go_forward(self):
raise NotImplementedError
def serialize(self):
"""Serialize into an opaque format understood by self.deserialize."""
raise NotImplementedError
def deserialize(self, data):
"""Serialize from a format produced by self.serialize."""
raise NotImplementedError
def load_items(self, items):
"""Deserialize from a list of WebHistoryItems."""
raise NotImplementedError
2016-06-13 11:50:58 +02:00
class AbstractTab(QWidget):
"""A wrapper over the given widget to hide its API and expose another one.
2016-06-13 14:44:41 +02:00
We use this to unify QWebView and QWebEngineView.
2016-06-13 15:05:31 +02:00
Attributes:
2016-06-14 11:03:34 +02:00
history: The AbstractHistory for the current tab.
2016-07-08 17:13:18 +02:00
registry: The ObjectRegistry associated with this tab.
2016-06-13 15:05:31 +02:00
_load_status: loading status of this page
Accessible via load_status() method.
_has_ssl_errors: Whether SSL errors happened.
Needs to be set by subclasses.
2016-06-13 15:05:31 +02:00
for properties, see WebView/WebEngineView docs.
2016-06-13 14:44:41 +02:00
Signals:
See related Qt signals.
2016-07-04 13:51:11 +02:00
new_tab_requested: Emitted when a new tab should be opened with the
given URL.
load_status_changed: The loading status changed
2016-06-13 11:50:58 +02:00
"""
2016-06-13 14:44:41 +02:00
window_close_requested = pyqtSignal()
link_hovered = pyqtSignal(str)
load_started = pyqtSignal()
load_progress = pyqtSignal(int)
load_finished = pyqtSignal(bool)
icon_changed = pyqtSignal(QIcon)
title_changed = pyqtSignal(str)
load_status_changed = pyqtSignal(str)
2016-07-04 13:51:11 +02:00
new_tab_requested = pyqtSignal(QUrl)
url_changed = pyqtSignal(QUrl)
2016-06-14 13:31:02 +02:00
shutting_down = pyqtSignal()
2016-07-27 15:34:28 +02:00
contents_size_changed = pyqtSignal(QSizeF)
2016-08-10 13:14:38 +02:00
add_history_item = pyqtSignal(QUrl, QUrl, str) # url, requested url, title
2016-06-13 14:44:41 +02:00
2016-06-14 15:20:40 +02:00
def __init__(self, win_id, parent=None):
self.win_id = win_id
2016-06-13 15:05:31 +02:00
self.tab_id = next(tab_id_gen)
2016-06-13 11:50:58 +02:00
super().__init__(parent)
2016-07-08 17:13:18 +02:00
self.registry = objreg.ObjectRegistry()
tab_registry = objreg.get('tab-registry', scope='window',
window=win_id)
tab_registry[self.tab_id] = self
objreg.register('tab', self, registry=self.registry)
2016-07-08 17:13:18 +02:00
2016-06-14 18:25:07 +02:00
# self.history = AbstractHistory(self)
# self.scroller = AbstractScroller(self, parent=self)
# self.caret = AbstractCaret(win_id=win_id, tab=self, mode_manager=...,
# parent=self)
2016-06-15 13:02:24 +02:00
# self.zoom = AbstractZoom(win_id=win_id)
2016-07-04 11:23:46 +02:00
# self.search = AbstractSearch(parent=self)
# self.printing = AbstractPrinting()
2016-08-17 19:20:14 +02:00
self.data = TabData()
self._layout = miscwidgets.WrapperLayout(self)
2016-06-13 15:05:31 +02:00
self._widget = None
self._progress = 0
self._has_ssl_errors = False
self._load_status = usertypes.LoadStatus.none
2016-08-10 16:37:52 +02:00
self._mouse_event_filter = mouse.MouseEventFilter(self, parent=self)
2016-07-04 15:35:08 +02:00
self.backend = None
2016-06-13 15:05:31 +02:00
# FIXME:qtwebengine Should this be public api via self.hints?
# Also, should we get it out of objreg?
hintmanager = hints.HintManager(win_id, self.tab_id, parent=self)
2016-08-17 19:20:14 +02:00
hintmanager.hint_events.connect(self._on_hint_events)
objreg.register('hintmanager', hintmanager, scope='tab',
window=self.win_id, tab=self.tab_id)
2016-06-13 15:05:31 +02:00
def _set_widget(self, widget):
2016-07-05 20:11:42 +02:00
# pylint: disable=protected-access
2016-06-13 11:50:58 +02:00
self._widget = widget
self._layout.wrap(self, widget)
2016-07-04 12:50:15 +02:00
self.history._history = widget.history()
self.scroller._init_widget(widget)
self.caret._widget = widget
self.zoom._widget = widget
self.search._widget = widget
self.printing._widget = widget
self._install_event_filter()
def _install_event_filter(self):
raise NotImplementedError
2016-06-13 14:44:41 +02:00
def _set_load_status(self, val):
"""Setter for load_status."""
if not isinstance(val, usertypes.LoadStatus):
raise TypeError("Type {} is no LoadStatus member!".format(val))
log.webview.debug("load status for {}: {}".format(repr(self), val))
self._load_status = val
self.load_status_changed.emit(val.name)
2016-08-17 18:02:24 +02:00
def post_event(self, evt):
"""Send the given event to the underlying widget."""
raise NotImplementedError
2016-08-17 19:20:14 +02:00
@pyqtSlot(usertypes.ClickTarget, list)
def _on_hint_events(self, target, events):
"""Post a new mouse event from a hintmanager."""
2016-08-17 19:20:14 +02:00
log.modes.debug("Sending hint events to {!r} with target {}".format(
self, target))
self._widget.setFocus()
2016-08-17 19:20:14 +02:00
self.data.hint_target = target
for evt in events:
self.post_event(evt)
def reset_target():
self.data.hint_target = None
QTimer.singleShot(0, reset_target)
@pyqtSlot(QUrl)
def _on_link_clicked(self, url):
log.webview.debug("link clicked: url {}, hint target {}, "
"open_target {}".format(
url.toDisplayString(),
self.data.hint_target, self.data.open_target))
if not url.isValid():
msg = urlutils.get_errstring(url, "Invalid link clicked")
message.error(self.win_id, msg)
self.data.open_target = usertypes.ClickTarget.normal
return False
target = self.data.combined_target()
if target == usertypes.ClickTarget.normal:
return
elif target == usertypes.ClickTarget.tab:
win_id = self.win_id
bg_tab = False
elif target == usertypes.ClickTarget.tab_bg:
win_id = self.win_id
bg_tab = True
elif target == usertypes.ClickTarget.window:
from qutebrowser.mainwindow import mainwindow
window = mainwindow.MainWindow()
window.show()
win_id = window.win_id
bg_tab = False
else:
raise ValueError("Invalid ClickTarget {}".format(target))
tabbed_browser = objreg.get('tabbed-browser', scope='window',
window=win_id)
tabbed_browser.tabopen(url, background=bg_tab)
self.data.open_target = usertypes.ClickTarget.normal
2016-07-11 14:36:57 +02:00
@pyqtSlot(QUrl)
def _on_url_changed(self, url):
"""Update title when URL has changed and no title is available."""
if url.isValid() and not self.title():
2016-07-11 16:32:37 +02:00
self.title_changed.emit(url.toDisplayString())
self.url_changed.emit(url)
2016-07-11 14:36:57 +02:00
2016-07-04 16:34:07 +02:00
@pyqtSlot()
def _on_load_started(self):
self._progress = 0
self._has_ssl_errors = False
2016-07-04 16:34:07 +02:00
self.data.viewing_source = False
self._set_load_status(usertypes.LoadStatus.loading)
2016-07-04 16:34:07 +02:00
self.load_started.emit()
@pyqtSlot(bool)
def _on_load_finished(self, ok):
if ok and not self._has_ssl_errors:
if self.url().scheme() == 'https':
self._set_load_status(usertypes.LoadStatus.success_https)
else:
self._set_load_status(usertypes.LoadStatus.success)
elif ok:
self._set_load_status(usertypes.LoadStatus.warn)
else:
self._set_load_status(usertypes.LoadStatus.error)
self.load_finished.emit(ok)
2016-07-11 14:36:57 +02:00
if not self.title():
self.title_changed.emit(self.url().toDisplayString())
2016-08-10 13:14:38 +02:00
@pyqtSlot()
def _on_history_trigger(self):
"""Emit add_history_item when triggered by backend-specific signal."""
url = self.url()
requested_url = self.url(requested=True)
self.add_history_item.emit(url, requested_url, self.title())
@pyqtSlot(int)
def _on_load_progress(self, perc):
self._progress = perc
self.load_progress.emit(perc)
@pyqtSlot()
def _on_ssl_errors(self):
self._has_ssl_errors = True
def url(self, requested=False):
2016-06-13 14:44:41 +02:00
raise NotImplementedError
def progress(self):
return self._progress
2016-06-13 14:44:41 +02:00
def load_status(self):
return self._load_status
2016-06-13 14:44:41 +02:00
def _openurl_prepare(self, url):
qtutils.ensure_valid(url)
self.title_changed.emit(url.toDisplayString())
2016-06-13 15:05:31 +02:00
def openurl(self, url):
raise NotImplementedError
2016-06-13 17:49:52 +02:00
2016-06-14 13:39:51 +02:00
def reload(self, *, force=False):
raise NotImplementedError
def stop(self):
raise NotImplementedError
def clear_ssl_errors(self):
raise NotImplementedError
2016-06-14 18:35:28 +02:00
def dump_async(self, callback, *, plain=False):
2016-06-14 13:26:30 +02:00
"""Dump the current page to a file ascync.
The given callback will be called with the result when dumping is
complete.
"""
raise NotImplementedError
2016-06-14 18:47:26 +02:00
def run_js_async(self, code, callback=None):
"""Run javascript async.
The given callback will be called with the result when running JS is
complete.
"""
raise NotImplementedError
2016-07-13 13:46:47 +02:00
def run_js_blocking(self, code):
"""Run javascript and block.
This returns the result to the caller. Its use should be avoided when
possible as it runs a local event loop for QtWebEngine.
"""
raise NotImplementedError
2016-06-14 13:31:02 +02:00
def shutdown(self):
raise NotImplementedError
2016-06-14 13:53:35 +02:00
def title(self):
raise NotImplementedError
2016-06-14 15:22:22 +02:00
def icon(self):
raise NotImplementedError
2016-07-04 16:33:49 +02:00
def set_html(self, html, base_url):
raise NotImplementedError
def find_all_elements(self, selector, callback, *, only_visible=False):
"""Find all HTML elements matching a given selector async.
2016-07-28 10:34:51 +02:00
Args:
callback: The callback to be called when the search finished.
2016-07-28 10:34:51 +02:00
selector: The CSS selector to search for.
only_visible: Only show elements which are visible on screen.
"""
raise NotImplementedError
def find_focus_element(self, callback):
"""Find the focused element on the page async.
Args:
callback: The callback to be called when the search finished.
Called with a WebEngineElement or None.
"""
raise NotImplementedError
def find_element_at_pos(self, pos, callback):
"""Find the element at the given position async.
This is also called "hit test" elsewhere.
Args:
pos: The QPoint to get the element for.
callback: The callback to be called when the search finished.
Called with a WebEngineElement or None.
"""
raise NotImplementedError
2016-06-13 17:49:52 +02:00
def __repr__(self):
try:
url = utils.elide(self.url().toDisplayString(QUrl.EncodeUnicode),
100)
except AttributeError:
url = '<AttributeError>'
2016-06-13 17:49:52 +02:00
return utils.get_repr(self, tab_id=self.tab_id, url=url)