1758 lines
68 KiB
Python
1758 lines
68 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/>.
|
|
|
|
"""Command dispatcher for TabbedBrowser."""
|
|
|
|
import os
|
|
import os.path
|
|
import sys
|
|
import shlex
|
|
import posixpath
|
|
import functools
|
|
import xml.etree.ElementTree
|
|
|
|
from PyQt5.QtWebKit import QWebSettings
|
|
from PyQt5.QtWidgets import QApplication, QTabBar
|
|
from PyQt5.QtCore import Qt, QUrl, QEvent
|
|
from PyQt5.QtGui import QClipboard, QKeyEvent
|
|
from PyQt5.QtPrintSupport import QPrintDialog, QPrintPreviewDialog
|
|
from PyQt5.QtWebKitWidgets import QWebPage
|
|
import pygments
|
|
import pygments.lexers
|
|
import pygments.formatters
|
|
|
|
from qutebrowser.commands import userscripts, cmdexc, cmdutils, runners
|
|
from qutebrowser.config import config, configexc
|
|
from qutebrowser.browser import webelem, inspector, urlmarks, downloads, mhtml
|
|
from qutebrowser.keyinput import modeman
|
|
from qutebrowser.utils import (message, usertypes, log, qtutils, urlutils,
|
|
objreg, utils)
|
|
from qutebrowser.utils.usertypes import KeyMode
|
|
from qutebrowser.misc import editor, guiprocess
|
|
|
|
|
|
class CommandDispatcher:
|
|
|
|
"""Command dispatcher for TabbedBrowser.
|
|
|
|
Contains all commands which are related to the current tab.
|
|
|
|
We can't simply add these commands to BrowserTab directly and use
|
|
currentWidget() for TabbedBrowser.cmd because at the time
|
|
cmdutils.register() decorators are run, currentWidget() will return None.
|
|
|
|
Attributes:
|
|
_editor: The ExternalEditor object.
|
|
_win_id: The window ID the CommandDispatcher is associated with.
|
|
_tabbed_browser: The TabbedBrowser used.
|
|
"""
|
|
|
|
def __init__(self, win_id, tabbed_browser):
|
|
self._editor = None
|
|
self._win_id = win_id
|
|
self._tabbed_browser = tabbed_browser
|
|
|
|
def __repr__(self):
|
|
return utils.get_repr(self)
|
|
|
|
def _new_tabbed_browser(self):
|
|
"""Get a tabbed-browser from a new window."""
|
|
from qutebrowser.mainwindow import mainwindow
|
|
new_window = mainwindow.MainWindow()
|
|
new_window.show()
|
|
return new_window.tabbed_browser
|
|
|
|
def _count(self):
|
|
"""Convenience method to get the widget count."""
|
|
return self._tabbed_browser.count()
|
|
|
|
def _set_current_index(self, idx):
|
|
"""Convenience method to set the current widget index."""
|
|
return self._tabbed_browser.setCurrentIndex(idx)
|
|
|
|
def _current_index(self):
|
|
"""Convenience method to get the current widget index."""
|
|
return self._tabbed_browser.currentIndex()
|
|
|
|
def _current_url(self):
|
|
"""Convenience method to get the current url."""
|
|
try:
|
|
return self._tabbed_browser.current_url()
|
|
except qtutils.QtValueError as e:
|
|
msg = "Current URL is invalid"
|
|
if e.reason:
|
|
msg += " ({})".format(e.reason)
|
|
msg += "!"
|
|
raise cmdexc.CommandError(msg)
|
|
|
|
def _current_title(self):
|
|
"""Convenience method to get the current title."""
|
|
return self._current_widget().title()
|
|
|
|
def _current_widget(self):
|
|
"""Get the currently active widget from a command."""
|
|
widget = self._tabbed_browser.currentWidget()
|
|
if widget is None:
|
|
raise cmdexc.CommandError("No WebView available yet!")
|
|
return widget
|
|
|
|
def _open(self, url, tab=False, background=False, window=False):
|
|
"""Helper function to open a page.
|
|
|
|
Args:
|
|
url: The URL to open as QUrl.
|
|
tab: Whether to open in a new tab.
|
|
background: Whether to open in the background.
|
|
window: Whether to open in a new window
|
|
"""
|
|
urlutils.raise_cmdexc_if_invalid(url)
|
|
tabbed_browser = self._tabbed_browser
|
|
cmdutils.check_exclusive((tab, background, window), 'tbw')
|
|
if window:
|
|
tabbed_browser = self._new_tabbed_browser()
|
|
tabbed_browser.tabopen(url)
|
|
elif tab:
|
|
tabbed_browser.tabopen(url, background=False, explicit=True)
|
|
elif background:
|
|
tabbed_browser.tabopen(url, background=True, explicit=True)
|
|
else:
|
|
widget = self._current_widget()
|
|
widget.openurl(url)
|
|
|
|
def _cntwidget(self, count=None):
|
|
"""Return a widget based on a count/idx.
|
|
|
|
Args:
|
|
count: The tab index, or None.
|
|
|
|
Return:
|
|
The current widget if count is None.
|
|
The widget with the given tab ID if count is given.
|
|
None if no widget was found.
|
|
"""
|
|
if count is None:
|
|
return self._tabbed_browser.currentWidget()
|
|
elif 1 <= count <= self._count():
|
|
cmdutils.check_overflow(count + 1, 'int')
|
|
return self._tabbed_browser.widget(count - 1)
|
|
else:
|
|
return None
|
|
|
|
def _tab_focus_last(self):
|
|
"""Select the tab which was last focused."""
|
|
try:
|
|
tab = objreg.get('last-focused-tab', scope='window',
|
|
window=self._win_id)
|
|
except KeyError:
|
|
raise cmdexc.CommandError("No last focused tab!")
|
|
idx = self._tabbed_browser.indexOf(tab)
|
|
if idx == -1:
|
|
raise cmdexc.CommandError("Last focused tab vanished!")
|
|
self._set_current_index(idx)
|
|
|
|
def _get_selection_override(self, left, right, opposite):
|
|
"""Helper function for tab_close to get the tab to select.
|
|
|
|
Args:
|
|
left: Force selecting the tab to the left of the current tab.
|
|
right: Force selecting the tab to the right of the current tab.
|
|
opposite: Force selecting the tab in the opposite direction of
|
|
what's configured in 'tabs->select-on-remove'.
|
|
|
|
Return:
|
|
QTabBar.SelectLeftTab, QTabBar.SelectRightTab, or None if no change
|
|
should be made.
|
|
"""
|
|
cmdutils.check_exclusive((left, right, opposite), 'lro')
|
|
if left:
|
|
return QTabBar.SelectLeftTab
|
|
elif right:
|
|
return QTabBar.SelectRightTab
|
|
elif opposite:
|
|
conf_selection = config.get('tabs', 'select-on-remove')
|
|
if conf_selection == QTabBar.SelectLeftTab:
|
|
return QTabBar.SelectRightTab
|
|
elif conf_selection == QTabBar.SelectRightTab:
|
|
return QTabBar.SelectLeftTab
|
|
elif conf_selection == QTabBar.SelectPreviousTab:
|
|
raise cmdexc.CommandError(
|
|
"-o is not supported with 'tabs->select-on-remove' set to "
|
|
"'previous'!")
|
|
else: # pragma: no cover
|
|
raise ValueError("Invalid select-on-remove value "
|
|
"{!r}!".format(conf_selection))
|
|
return None
|
|
|
|
@cmdutils.register(instance='command-dispatcher', scope='window',
|
|
count='count')
|
|
def tab_close(self, left=False, right=False, opposite=False, count=None):
|
|
"""Close the current/[count]th tab.
|
|
|
|
Args:
|
|
left: Force selecting the tab to the left of the current tab.
|
|
right: Force selecting the tab to the right of the current tab.
|
|
opposite: Force selecting the tab in the opposite direction of
|
|
what's configured in 'tabs->select-on-remove'.
|
|
count: The tab index to close, or None
|
|
"""
|
|
tab = self._cntwidget(count)
|
|
if tab is None:
|
|
return
|
|
tabbar = self._tabbed_browser.tabBar()
|
|
selection_override = self._get_selection_override(left, right,
|
|
opposite)
|
|
if selection_override is None:
|
|
self._tabbed_browser.close_tab(tab)
|
|
else:
|
|
old_selection_behavior = tabbar.selectionBehaviorOnRemove()
|
|
tabbar.setSelectionBehaviorOnRemove(selection_override)
|
|
self._tabbed_browser.close_tab(tab)
|
|
tabbar.setSelectionBehaviorOnRemove(old_selection_behavior)
|
|
|
|
@cmdutils.register(instance='command-dispatcher', name='open',
|
|
maxsplit=0, scope='window', count='count',
|
|
completion=[usertypes.Completion.url])
|
|
def openurl(self, url=None, bg=False, tab=False, window=False, count=None):
|
|
"""Open a URL in the current/[count]th tab.
|
|
|
|
Args:
|
|
url: The URL to open.
|
|
bg: Open in a new background tab.
|
|
tab: Open in a new tab.
|
|
window: Open in a new window.
|
|
count: The tab index to open the URL in, or None.
|
|
"""
|
|
if url is None:
|
|
if tab or bg or window:
|
|
url = config.get('general', 'default-page')
|
|
else:
|
|
raise cmdexc.CommandError("No URL given, but -t/-b/-w is not "
|
|
"set!")
|
|
else:
|
|
try:
|
|
url = urlutils.fuzzy_url(url)
|
|
except urlutils.InvalidUrlError as e:
|
|
raise cmdexc.CommandError(e)
|
|
if tab or bg or window:
|
|
self._open(url, tab, bg, window)
|
|
else:
|
|
curtab = self._cntwidget(count)
|
|
if curtab is None:
|
|
if count is None:
|
|
# We want to open a URL in the current tab, but none exists
|
|
# yet.
|
|
self._tabbed_browser.tabopen(url)
|
|
else:
|
|
# Explicit count with a tab that doesn't exist.
|
|
return
|
|
else:
|
|
curtab.openurl(url)
|
|
|
|
@cmdutils.register(instance='command-dispatcher', name='reload',
|
|
scope='window', count='count')
|
|
def reloadpage(self, force=False, count=None):
|
|
"""Reload the current/[count]th tab.
|
|
|
|
Args:
|
|
count: The tab index to reload, or None.
|
|
force: Bypass the page cache.
|
|
"""
|
|
tab = self._cntwidget(count)
|
|
if tab is not None:
|
|
if force:
|
|
tab.page().triggerAction(QWebPage.ReloadAndBypassCache)
|
|
else:
|
|
tab.reload()
|
|
|
|
@cmdutils.register(instance='command-dispatcher', scope='window',
|
|
count='count')
|
|
def stop(self, count=None):
|
|
"""Stop loading in the current/[count]th tab.
|
|
|
|
Args:
|
|
count: The tab index to stop, or None.
|
|
"""
|
|
tab = self._cntwidget(count)
|
|
if tab is not None:
|
|
tab.stop()
|
|
|
|
@cmdutils.register(instance='command-dispatcher', name='print',
|
|
scope='window', count='count')
|
|
def printpage(self, preview=False, count=None):
|
|
"""Print the current/[count]th tab.
|
|
|
|
Args:
|
|
preview: Show preview instead of printing.
|
|
count: The tab index to print, or None.
|
|
"""
|
|
if not qtutils.check_print_compat():
|
|
# WORKAROUND (remove this when we bump the requirements to 5.3.0)
|
|
raise cmdexc.CommandError(
|
|
"Printing on Qt < 5.3.0 on Windows is broken, please upgrade!")
|
|
tab = self._cntwidget(count)
|
|
if tab is not None:
|
|
if preview:
|
|
diag = QPrintPreviewDialog()
|
|
diag.setAttribute(Qt.WA_DeleteOnClose)
|
|
diag.setWindowFlags(diag.windowFlags() |
|
|
Qt.WindowMaximizeButtonHint |
|
|
Qt.WindowMinimizeButtonHint)
|
|
diag.paintRequested.connect(tab.print)
|
|
diag.exec_()
|
|
else:
|
|
diag = QPrintDialog()
|
|
diag.setAttribute(Qt.WA_DeleteOnClose)
|
|
diag.open(lambda: tab.print(diag.printer()))
|
|
|
|
@cmdutils.register(instance='command-dispatcher', scope='window')
|
|
def tab_clone(self, bg=False, window=False):
|
|
"""Duplicate the current tab.
|
|
|
|
Args:
|
|
bg: Open in a background tab.
|
|
window: Open in a new window.
|
|
|
|
Return:
|
|
The new QWebView.
|
|
"""
|
|
cmdutils.check_exclusive((bg, window), 'bw')
|
|
curtab = self._current_widget()
|
|
cur_title = self._tabbed_browser.page_title(self._current_index())
|
|
# The new tab could be in a new tabbed_browser (e.g. because of
|
|
# tabs-are-windows being set)
|
|
if window:
|
|
new_tabbed_browser = self._new_tabbed_browser()
|
|
else:
|
|
new_tabbed_browser = self._tabbed_browser
|
|
newtab = new_tabbed_browser.tabopen(background=bg, explicit=True)
|
|
new_tabbed_browser = objreg.get('tabbed-browser', scope='window',
|
|
window=newtab.win_id)
|
|
idx = new_tabbed_browser.indexOf(newtab)
|
|
new_tabbed_browser.set_page_title(idx, cur_title)
|
|
if config.get('tabs', 'show-favicons'):
|
|
new_tabbed_browser.setTabIcon(idx, curtab.icon())
|
|
newtab.keep_icon = True
|
|
newtab.setZoomFactor(curtab.zoomFactor())
|
|
history = qtutils.serialize(curtab.history())
|
|
qtutils.deserialize(history, newtab.history())
|
|
return newtab
|
|
|
|
@cmdutils.register(instance='command-dispatcher', scope='window')
|
|
def tab_detach(self):
|
|
"""Detach the current tab to its own window."""
|
|
url = self._current_url()
|
|
self._open(url, window=True)
|
|
cur_widget = self._current_widget()
|
|
self._tabbed_browser.close_tab(cur_widget)
|
|
|
|
def _back_forward(self, tab, bg, window, count, forward):
|
|
"""Helper function for :back/:forward."""
|
|
# Catch common cases before e.g. cloning tab
|
|
history = self._current_widget().page().history()
|
|
if not forward and not history.canGoBack():
|
|
raise cmdexc.CommandError("At beginning of history.")
|
|
elif forward and not history.canGoForward():
|
|
raise cmdexc.CommandError("At end of history.")
|
|
|
|
if tab or bg or window:
|
|
widget = self.tab_clone(bg, window)
|
|
else:
|
|
widget = self._current_widget()
|
|
|
|
history = widget.page().history()
|
|
for _ in range(count):
|
|
if forward:
|
|
if not history.canGoForward():
|
|
raise cmdexc.CommandError("At end of history.")
|
|
widget.forward()
|
|
else:
|
|
if not history.canGoBack():
|
|
raise cmdexc.CommandError("At beginning of history.")
|
|
widget.back()
|
|
|
|
@cmdutils.register(instance='command-dispatcher', scope='window',
|
|
count='count')
|
|
def back(self, tab=False, bg=False, window=False, count=1):
|
|
"""Go back in the history of the current tab.
|
|
|
|
Args:
|
|
tab: Go back in a new tab.
|
|
bg: Go back in a background tab.
|
|
window: Go back in a new window.
|
|
count: How many pages to go back.
|
|
"""
|
|
self._back_forward(tab, bg, window, count, forward=False)
|
|
|
|
@cmdutils.register(instance='command-dispatcher', scope='window',
|
|
count='count')
|
|
def forward(self, tab=False, bg=False, window=False, count=1):
|
|
"""Go forward in the history of the current tab.
|
|
|
|
Args:
|
|
tab: Go forward in a new tab.
|
|
bg: Go forward in a background tab.
|
|
window: Go forward in a new window.
|
|
count: How many pages to go forward.
|
|
"""
|
|
self._back_forward(tab, bg, window, count, forward=True)
|
|
|
|
def _navigate_incdec(self, url, incdec, tab, background, window):
|
|
"""Helper method for :navigate when `where' is increment/decrement.
|
|
|
|
Args:
|
|
url: The current url.
|
|
incdec: Either 'increment' or 'decrement'.
|
|
tab: Whether to open the link in a new tab.
|
|
background: Open the link in a new background tab.
|
|
window: Open the link in a new window.
|
|
"""
|
|
segments = set(config.get('general', 'url-incdec-segments'))
|
|
try:
|
|
new_url = urlutils.incdec_number(url, incdec, segments=segments)
|
|
except urlutils.IncDecError as error:
|
|
raise cmdexc.CommandError(error.msg)
|
|
self._open(new_url, tab, background, window)
|
|
|
|
def _navigate_up(self, url, tab, background, window):
|
|
"""Helper method for :navigate when `where' is up.
|
|
|
|
Args:
|
|
url: The current url.
|
|
tab: Whether to open the link in a new tab.
|
|
background: Open the link in a new background tab.
|
|
window: Open the link in a new window.
|
|
"""
|
|
path = url.path()
|
|
if not path or path == '/':
|
|
raise cmdexc.CommandError("Can't go up!")
|
|
new_path = posixpath.join(path, posixpath.pardir)
|
|
url.setPath(new_path)
|
|
self._open(url, tab, background, window)
|
|
|
|
@cmdutils.register(instance='command-dispatcher', scope='window')
|
|
def navigate(self, where: {'type': ('prev', 'next', 'up', 'increment',
|
|
'decrement')},
|
|
tab=False, bg=False, window=False):
|
|
"""Open typical prev/next links or navigate using the URL path.
|
|
|
|
This tries to automatically click on typical _Previous Page_ or
|
|
_Next Page_ links using some heuristics.
|
|
|
|
Alternatively it can navigate by changing the current URL.
|
|
|
|
Args:
|
|
where: What to open.
|
|
|
|
- `prev`: Open a _previous_ link.
|
|
- `next`: Open a _next_ link.
|
|
- `up`: Go up a level in the current URL.
|
|
- `increment`: Increment the last number in the URL.
|
|
- `decrement`: Decrement the last number in the URL.
|
|
|
|
tab: Open in a new tab.
|
|
bg: Open in a background tab.
|
|
window: Open in a new window.
|
|
"""
|
|
cmdutils.check_exclusive((tab, bg, window), 'tbw')
|
|
widget = self._current_widget()
|
|
frame = widget.page().currentFrame()
|
|
url = self._current_url()
|
|
if frame is None:
|
|
raise cmdexc.CommandError("No frame focused!")
|
|
hintmanager = objreg.get('hintmanager', scope='tab', tab='current')
|
|
if where == 'prev':
|
|
hintmanager.follow_prevnext(frame, url, prev=True, tab=tab,
|
|
background=bg, window=window)
|
|
elif where == 'next':
|
|
hintmanager.follow_prevnext(frame, url, prev=False, tab=tab,
|
|
background=bg, window=window)
|
|
elif where == 'up':
|
|
self._navigate_up(url, tab, bg, window)
|
|
elif where in ('decrement', 'increment'):
|
|
self._navigate_incdec(url, where, tab, bg, window)
|
|
else: # pragma: no cover
|
|
raise ValueError("Got called with invalid value {} for "
|
|
"`where'.".format(where))
|
|
|
|
@cmdutils.register(instance='command-dispatcher', hide=True,
|
|
scope='window', count='count')
|
|
def scroll_px(self, dx: {'type': int}, dy: {'type': int}, count=1):
|
|
"""Scroll the current tab by 'count * dx/dy' pixels.
|
|
|
|
Args:
|
|
dx: How much to scroll in x-direction.
|
|
dy: How much to scroll in x-direction.
|
|
count: multiplier
|
|
"""
|
|
dx *= count
|
|
dy *= count
|
|
cmdutils.check_overflow(dx, 'int')
|
|
cmdutils.check_overflow(dy, 'int')
|
|
self._current_widget().page().currentFrame().scroll(dx, dy)
|
|
|
|
@cmdutils.register(instance='command-dispatcher', hide=True,
|
|
scope='window', count='count')
|
|
def scroll(self, direction: {'type': (str, int)}, count=1):
|
|
"""Scroll the current tab in the given direction.
|
|
|
|
Args:
|
|
direction: In which direction to scroll
|
|
(up/down/left/right/top/bottom).
|
|
count: multiplier
|
|
"""
|
|
fake_keys = {
|
|
'up': Qt.Key_Up,
|
|
'down': Qt.Key_Down,
|
|
'left': Qt.Key_Left,
|
|
'right': Qt.Key_Right,
|
|
'top': Qt.Key_Home,
|
|
'bottom': Qt.Key_End,
|
|
'page-up': Qt.Key_PageUp,
|
|
'page-down': Qt.Key_PageDown,
|
|
}
|
|
try:
|
|
key = fake_keys[direction]
|
|
except KeyError:
|
|
expected_values = ', '.join(sorted(fake_keys))
|
|
raise cmdexc.CommandError("Invalid value {!r} for direction - "
|
|
"expected one of: {}".format(
|
|
direction, expected_values))
|
|
widget = self._current_widget()
|
|
frame = widget.page().currentFrame()
|
|
|
|
press_evt = QKeyEvent(QEvent.KeyPress, key, Qt.NoModifier, 0, 0, 0)
|
|
release_evt = QKeyEvent(QEvent.KeyRelease, key, Qt.NoModifier, 0, 0, 0)
|
|
|
|
# Count doesn't make sense with top/bottom
|
|
if direction in ('top', 'bottom'):
|
|
count = 1
|
|
|
|
max_min = {
|
|
'up': [Qt.Vertical, frame.scrollBarMinimum],
|
|
'down': [Qt.Vertical, frame.scrollBarMaximum],
|
|
'left': [Qt.Horizontal, frame.scrollBarMinimum],
|
|
'right': [Qt.Horizontal, frame.scrollBarMaximum],
|
|
'page-up': [Qt.Vertical, frame.scrollBarMinimum],
|
|
'page-down': [Qt.Vertical, frame.scrollBarMaximum],
|
|
}
|
|
|
|
for _ in range(count):
|
|
# Abort scrolling if the minimum/maximum was reached.
|
|
try:
|
|
qt_dir, getter = max_min[direction]
|
|
except KeyError:
|
|
pass
|
|
else:
|
|
if frame.scrollBarValue(qt_dir) == getter(qt_dir):
|
|
return
|
|
|
|
widget.keyPressEvent(press_evt)
|
|
widget.keyReleaseEvent(release_evt)
|
|
|
|
@cmdutils.register(instance='command-dispatcher', hide=True,
|
|
scope='window', count='count')
|
|
def scroll_perc(self, perc: {'type': float}=None,
|
|
horizontal: {'flag': 'x'}=False, count=None):
|
|
"""Scroll to a specific percentage of the page.
|
|
|
|
The percentage can be given either as argument or as count.
|
|
If no percentage is given, the page is scrolled to the end.
|
|
|
|
Args:
|
|
perc: Percentage to scroll.
|
|
horizontal: Scroll horizontally instead of vertically.
|
|
count: Percentage to scroll.
|
|
"""
|
|
if perc is None and count is None:
|
|
perc = 100
|
|
elif perc is None:
|
|
perc = count
|
|
|
|
orientation = Qt.Horizontal if horizontal else Qt.Vertical
|
|
|
|
if perc == 0 and orientation == Qt.Vertical:
|
|
self.scroll('top')
|
|
elif perc == 100 and orientation == Qt.Vertical:
|
|
self.scroll('bottom')
|
|
else:
|
|
perc = qtutils.check_overflow(perc, 'int', fatal=False)
|
|
frame = self._current_widget().page().currentFrame()
|
|
m = frame.scrollBarMaximum(orientation)
|
|
if m == 0:
|
|
return
|
|
frame.setScrollBarValue(orientation, int(m * perc / 100))
|
|
|
|
@cmdutils.register(instance='command-dispatcher', hide=True,
|
|
scope='window', count='count')
|
|
def scroll_page(self, x: {'type': float}, y: {'type': float}, *,
|
|
top_navigate: {'type': ('prev', 'decrement'),
|
|
'metavar': 'ACTION'}=None,
|
|
bottom_navigate: {'type': ('next', 'increment'),
|
|
'metavar': 'ACTION'}=None,
|
|
count=1):
|
|
"""Scroll the frame page-wise.
|
|
|
|
Args:
|
|
x: How many pages to scroll to the right.
|
|
y: How many pages to scroll down.
|
|
bottom_navigate: :navigate action (next, increment) to run when
|
|
scrolling down at the bottom of the page.
|
|
top_navigate: :navigate action (prev, decrement) to run when
|
|
scrolling up at the top of the page.
|
|
count: multiplier
|
|
"""
|
|
frame = self._current_widget().page().currentFrame()
|
|
if not frame.url().isValid():
|
|
# See https://github.com/The-Compiler/qutebrowser/issues/701
|
|
return
|
|
|
|
if (bottom_navigate is not None and
|
|
frame.scrollPosition().y() >=
|
|
frame.scrollBarMaximum(Qt.Vertical)):
|
|
self.navigate(bottom_navigate)
|
|
return
|
|
elif top_navigate is not None and frame.scrollPosition().y() == 0:
|
|
self.navigate(top_navigate)
|
|
return
|
|
|
|
mult_x = count * x
|
|
mult_y = count * y
|
|
if mult_y.is_integer():
|
|
if mult_y == 0:
|
|
pass
|
|
elif mult_y < 0:
|
|
self.scroll('page-up', count=-int(mult_y))
|
|
elif mult_y > 0: # pragma: no branch
|
|
self.scroll('page-down', count=int(mult_y))
|
|
mult_y = 0
|
|
if mult_x == 0 and mult_y == 0:
|
|
return
|
|
size = frame.geometry()
|
|
dx = mult_x * size.width()
|
|
dy = mult_y * size.height()
|
|
cmdutils.check_overflow(dx, 'int')
|
|
cmdutils.check_overflow(dy, 'int')
|
|
frame.scroll(dx, dy)
|
|
|
|
@cmdutils.register(instance='command-dispatcher', scope='window')
|
|
def yank(self, title=False, sel=False, domain=False):
|
|
"""Yank the current URL/title to the clipboard or primary selection.
|
|
|
|
Args:
|
|
sel: Use the primary selection instead of the clipboard.
|
|
title: Yank the title instead of the URL.
|
|
domain: Yank only the scheme, domain, and port number.
|
|
"""
|
|
clipboard = QApplication.clipboard()
|
|
if title:
|
|
s = self._tabbed_browser.page_title(self._current_index())
|
|
what = 'title'
|
|
elif domain:
|
|
port = self._current_url().port()
|
|
s = '{}://{}{}'.format(self._current_url().scheme(),
|
|
self._current_url().host(),
|
|
':' + str(port) if port > -1 else '')
|
|
what = 'domain'
|
|
else:
|
|
s = self._current_url().toString(
|
|
QUrl.FullyEncoded | QUrl.RemovePassword)
|
|
what = 'URL'
|
|
if sel and clipboard.supportsSelection():
|
|
mode = QClipboard.Selection
|
|
target = "primary selection"
|
|
else:
|
|
mode = QClipboard.Clipboard
|
|
target = "clipboard"
|
|
log.misc.debug("Yanking to {}: '{}'".format(target, s))
|
|
|
|
msg = "Yanked {} to {}: {}".format(what, target, s)
|
|
clipboard.changed.connect(functools.partial(
|
|
self._display_yank_msg, clipboard, msg))
|
|
clipboard.setText(s, mode)
|
|
|
|
def _display_yank_msg(self, clipboard, msg):
|
|
"""Display a message when something was yanked."""
|
|
message.info(self._win_id, msg)
|
|
clipboard.changed.disconnect()
|
|
|
|
@cmdutils.register(instance='command-dispatcher', scope='window',
|
|
count='count')
|
|
def zoom_in(self, count=1):
|
|
"""Increase the zoom level for the current tab.
|
|
|
|
Args:
|
|
count: How many steps to zoom in.
|
|
"""
|
|
tab = self._current_widget()
|
|
try:
|
|
perc = tab.zoom(count)
|
|
except ValueError as e:
|
|
raise cmdexc.CommandError(e)
|
|
message.info(self._win_id, "Zoom level: {}%".format(perc))
|
|
|
|
@cmdutils.register(instance='command-dispatcher', scope='window',
|
|
count='count')
|
|
def zoom_out(self, count=1):
|
|
"""Decrease the zoom level for the current tab.
|
|
|
|
Args:
|
|
count: How many steps to zoom out.
|
|
"""
|
|
tab = self._current_widget()
|
|
try:
|
|
perc = tab.zoom(-count)
|
|
except ValueError as e:
|
|
raise cmdexc.CommandError(e)
|
|
message.info(self._win_id, "Zoom level: {}%".format(perc))
|
|
|
|
@cmdutils.register(instance='command-dispatcher', scope='window',
|
|
count='count')
|
|
def zoom(self, zoom: {'type': int}=None, count=None):
|
|
"""Set the zoom level for the current tab.
|
|
|
|
The zoom can be given as argument or as [count]. If neither of both is
|
|
given, the zoom is set to the default zoom.
|
|
|
|
Args:
|
|
zoom: The zoom percentage to set.
|
|
count: The zoom percentage to set.
|
|
"""
|
|
try:
|
|
default = config.get('ui', 'default-zoom')
|
|
level = cmdutils.arg_or_count(zoom, count, default=default)
|
|
except ValueError as e:
|
|
raise cmdexc.CommandError(e)
|
|
tab = self._current_widget()
|
|
|
|
try:
|
|
tab.zoom_perc(level)
|
|
except ValueError as e:
|
|
raise cmdexc.CommandError(e)
|
|
message.info(self._win_id, "Zoom level: {}%".format(level))
|
|
|
|
@cmdutils.register(instance='command-dispatcher', scope='window')
|
|
def tab_only(self, left=False, right=False):
|
|
"""Close all tabs except for the current one.
|
|
|
|
Args:
|
|
left: Keep tabs to the left of the current.
|
|
right: Keep tabs to the right of the current.
|
|
"""
|
|
cmdutils.check_exclusive((left, right), 'lr')
|
|
cur_idx = self._tabbed_browser.currentIndex()
|
|
assert cur_idx != -1
|
|
|
|
for i, tab in enumerate(self._tabbed_browser.widgets()):
|
|
if (i == cur_idx or (left and i < cur_idx) or
|
|
(right and i > cur_idx)):
|
|
continue
|
|
else:
|
|
self._tabbed_browser.close_tab(tab)
|
|
|
|
@cmdutils.register(instance='command-dispatcher', scope='window')
|
|
def undo(self):
|
|
"""Re-open a closed tab (optionally skipping [count] closed tabs)."""
|
|
try:
|
|
self._tabbed_browser.undo()
|
|
except IndexError:
|
|
raise cmdexc.CommandError("Nothing to undo!")
|
|
|
|
@cmdutils.register(instance='command-dispatcher', scope='window',
|
|
count='count')
|
|
def tab_prev(self, count=1):
|
|
"""Switch to the previous tab, or switch [count] tabs back.
|
|
|
|
Args:
|
|
count: How many tabs to switch back.
|
|
"""
|
|
newidx = self._current_index() - count
|
|
if newidx >= 0:
|
|
self._set_current_index(newidx)
|
|
elif config.get('tabs', 'wrap'):
|
|
self._set_current_index(newidx % self._count())
|
|
else:
|
|
raise cmdexc.CommandError("First tab")
|
|
|
|
@cmdutils.register(instance='command-dispatcher', scope='window',
|
|
count='count')
|
|
def tab_next(self, count=1):
|
|
"""Switch to the next tab, or switch [count] tabs forward.
|
|
|
|
Args:
|
|
count: How many tabs to switch forward.
|
|
"""
|
|
newidx = self._current_index() + count
|
|
if newidx < self._count():
|
|
self._set_current_index(newidx)
|
|
elif config.get('tabs', 'wrap'):
|
|
self._set_current_index(newidx % self._count())
|
|
else:
|
|
raise cmdexc.CommandError("Last tab")
|
|
|
|
@cmdutils.register(instance='command-dispatcher', scope='window')
|
|
def paste(self, sel=False, tab=False, bg=False, window=False):
|
|
"""Open a page from the clipboard.
|
|
|
|
Args:
|
|
sel: Use the primary selection instead of the clipboard.
|
|
tab: Open in a new tab.
|
|
bg: Open in a background tab.
|
|
window: Open in new window.
|
|
"""
|
|
clipboard = QApplication.clipboard()
|
|
if sel and clipboard.supportsSelection():
|
|
mode = QClipboard.Selection
|
|
target = "Primary selection"
|
|
else:
|
|
mode = QClipboard.Clipboard
|
|
target = "Clipboard"
|
|
text = clipboard.text(mode)
|
|
if not text:
|
|
raise cmdexc.CommandError("{} is empty.".format(target))
|
|
log.misc.debug("{} contained: '{}'".format(target, text))
|
|
try:
|
|
url = urlutils.fuzzy_url(text)
|
|
except urlutils.InvalidUrlError as e:
|
|
raise cmdexc.CommandError(e)
|
|
self._open(url, tab, bg, window)
|
|
|
|
@cmdutils.register(instance='command-dispatcher', scope='window',
|
|
count='count')
|
|
def tab_focus(self, index: {'type': (int, 'last')}=None, count=None):
|
|
"""Select the tab given as argument/[count].
|
|
|
|
If neither count nor index are given, it behaves like tab-next.
|
|
|
|
Args:
|
|
index: The tab index to focus, starting with 1. The special value
|
|
`last` focuses the last focused tab.
|
|
count: The tab index to focus, starting with 1.
|
|
"""
|
|
if index == 'last':
|
|
self._tab_focus_last()
|
|
return
|
|
if index is None and count is None:
|
|
self.tab_next()
|
|
return
|
|
try:
|
|
idx = cmdutils.arg_or_count(index, count, default=1,
|
|
countzero=self._count())
|
|
except ValueError as e:
|
|
raise cmdexc.CommandError(e)
|
|
cmdutils.check_overflow(idx + 1, 'int')
|
|
if 1 <= idx <= self._count():
|
|
self._set_current_index(idx - 1)
|
|
else:
|
|
raise cmdexc.CommandError("There's no tab with index {}!".format(
|
|
idx))
|
|
|
|
@cmdutils.register(instance='command-dispatcher', scope='window',
|
|
count='count')
|
|
def tab_move(self, direction: {'type': ('+', '-')}=None, count=None):
|
|
"""Move the current tab.
|
|
|
|
Args:
|
|
direction: `+` or `-` for relative moving, not given for absolute
|
|
moving.
|
|
count: If moving absolutely: New position (default: 0)
|
|
If moving relatively: Offset.
|
|
"""
|
|
if direction is None:
|
|
# absolute moving
|
|
new_idx = 0 if count is None else count - 1
|
|
elif direction in '+-':
|
|
# relative moving
|
|
delta = 1 if count is None else count
|
|
if direction == '-':
|
|
new_idx = self._current_index() - delta
|
|
elif direction == '+': # pragma: no branch
|
|
new_idx = self._current_index() + delta
|
|
|
|
if config.get('tabs', 'wrap'):
|
|
new_idx %= self._count()
|
|
else: # pragma: no cover
|
|
raise ValueError("Invalid direction '{}'!".format(direction))
|
|
|
|
if not 0 <= new_idx < self._count():
|
|
raise cmdexc.CommandError("Can't move tab to position {}!".format(
|
|
new_idx + 1))
|
|
|
|
tab = self._current_widget()
|
|
cur_idx = self._current_index()
|
|
icon = self._tabbed_browser.tabIcon(cur_idx)
|
|
label = self._tabbed_browser.page_title(cur_idx)
|
|
cmdutils.check_overflow(cur_idx, 'int')
|
|
cmdutils.check_overflow(new_idx, 'int')
|
|
self._tabbed_browser.setUpdatesEnabled(False)
|
|
try:
|
|
self._tabbed_browser.removeTab(cur_idx)
|
|
self._tabbed_browser.insertTab(new_idx, tab, icon, label)
|
|
self._set_current_index(new_idx)
|
|
finally:
|
|
self._tabbed_browser.setUpdatesEnabled(True)
|
|
|
|
@cmdutils.register(instance='command-dispatcher', scope='window',
|
|
maxsplit=0)
|
|
def spawn(self, cmdline, userscript=False, verbose=False, detach=False):
|
|
"""Spawn a command in a shell.
|
|
|
|
Note the {url} variable which gets replaced by the current URL might be
|
|
useful here.
|
|
|
|
Args:
|
|
userscript: Run the command as a userscript. Either store the
|
|
userscript in `~/.local/share/qutebrowser/userscripts`
|
|
(or `$XDG_DATA_DIR`), or use an absolute path.
|
|
verbose: Show notifications when the command started/exited.
|
|
detach: Whether the command should be detached from qutebrowser.
|
|
cmdline: The commandline to execute.
|
|
"""
|
|
try:
|
|
cmd, *args = shlex.split(cmdline)
|
|
except ValueError as e:
|
|
raise cmdexc.CommandError("Error while splitting command: "
|
|
"{}".format(e))
|
|
|
|
args = runners.replace_variables(self._win_id, args)
|
|
|
|
log.procs.debug("Executing {} with args {}, userscript={}".format(
|
|
cmd, args, userscript))
|
|
if userscript:
|
|
# ~ expansion is handled by the userscript module.
|
|
self._run_userscript(cmd, *args, verbose=verbose)
|
|
else:
|
|
cmd = os.path.expanduser(cmd)
|
|
proc = guiprocess.GUIProcess(self._win_id, what='command',
|
|
verbose=verbose,
|
|
parent=self._tabbed_browser)
|
|
if detach:
|
|
proc.start_detached(cmd, args)
|
|
else:
|
|
proc.start(cmd, args)
|
|
|
|
@cmdutils.register(instance='command-dispatcher', scope='window')
|
|
def home(self):
|
|
"""Open main startpage in current tab."""
|
|
self.openurl(config.get('general', 'startpage')[0])
|
|
|
|
def _run_userscript(self, cmd, *args, verbose=False):
|
|
"""Run a userscript given as argument.
|
|
|
|
Args:
|
|
cmd: The userscript to run.
|
|
args: Arguments to pass to the userscript.
|
|
verbose: Show notifications when the command started/exited.
|
|
"""
|
|
env = {
|
|
'QUTE_MODE': 'command',
|
|
}
|
|
|
|
idx = self._current_index()
|
|
if idx != -1:
|
|
env['QUTE_TITLE'] = self._tabbed_browser.page_title(idx)
|
|
|
|
webview = self._tabbed_browser.currentWidget()
|
|
if webview is None:
|
|
mainframe = None
|
|
else:
|
|
if webview.hasSelection():
|
|
env['QUTE_SELECTED_TEXT'] = webview.selectedText()
|
|
env['QUTE_SELECTED_HTML'] = webview.selectedHtml()
|
|
mainframe = webview.page().mainFrame()
|
|
|
|
try:
|
|
url = self._tabbed_browser.current_url()
|
|
except qtutils.QtValueError:
|
|
pass
|
|
else:
|
|
env['QUTE_URL'] = url.toString(QUrl.FullyEncoded)
|
|
|
|
env.update(userscripts.store_source(mainframe))
|
|
userscripts.run(cmd, *args, win_id=self._win_id, env=env,
|
|
verbose=verbose)
|
|
|
|
@cmdutils.register(instance='command-dispatcher', scope='window')
|
|
def quickmark_save(self):
|
|
"""Save the current page as a quickmark."""
|
|
quickmark_manager = objreg.get('quickmark-manager')
|
|
quickmark_manager.prompt_save(self._win_id, self._current_url())
|
|
|
|
@cmdutils.register(instance='command-dispatcher', scope='window',
|
|
maxsplit=0,
|
|
completion=[usertypes.Completion.quickmark_by_name])
|
|
def quickmark_load(self, name, tab=False, bg=False, window=False):
|
|
"""Load a quickmark.
|
|
|
|
Args:
|
|
name: The name of the quickmark to load.
|
|
tab: Load the quickmark in a new tab.
|
|
bg: Load the quickmark in a new background tab.
|
|
window: Load the quickmark in a new window.
|
|
"""
|
|
try:
|
|
url = objreg.get('quickmark-manager').get(name)
|
|
except urlmarks.Error as e:
|
|
raise cmdexc.CommandError(str(e))
|
|
self._open(url, tab, bg, window)
|
|
|
|
@cmdutils.register(instance='command-dispatcher', scope='window')
|
|
def bookmark_add(self):
|
|
"""Save the current page as a bookmark."""
|
|
bookmark_manager = objreg.get('bookmark-manager')
|
|
url = self._current_url()
|
|
try:
|
|
bookmark_manager.add(url, self._current_title())
|
|
except urlmarks.Error as e:
|
|
raise cmdexc.CommandError(str(e))
|
|
else:
|
|
message.info(self._win_id,
|
|
"Bookmarked {}!".format(url.toDisplayString()))
|
|
|
|
@cmdutils.register(instance='command-dispatcher', scope='window',
|
|
maxsplit=0,
|
|
completion=[usertypes.Completion.bookmark_by_url])
|
|
def bookmark_load(self, url, tab=False, bg=False, window=False):
|
|
"""Load a bookmark.
|
|
|
|
Args:
|
|
url: The url of the bookmark to load.
|
|
tab: Load the bookmark in a new tab.
|
|
bg: Load the bookmark in a new background tab.
|
|
window: Load the bookmark in a new window.
|
|
"""
|
|
try:
|
|
url = urlutils.fuzzy_url(url)
|
|
except urlutils.InvalidUrlError as e:
|
|
raise cmdexc.CommandError(e)
|
|
self._open(url, tab, bg, window)
|
|
|
|
@cmdutils.register(instance='command-dispatcher', hide=True,
|
|
scope='window')
|
|
def follow_selected(self, tab=False):
|
|
"""Follow the selected text.
|
|
|
|
Args:
|
|
tab: Load the selected link in a new tab.
|
|
"""
|
|
widget = self._current_widget()
|
|
page = widget.page()
|
|
if not page.hasSelection():
|
|
return
|
|
if QWebSettings.globalSettings().testAttribute(
|
|
QWebSettings.JavascriptEnabled):
|
|
if tab:
|
|
page.open_target = usertypes.ClickTarget.tab
|
|
page.currentFrame().evaluateJavaScript(
|
|
'window.getSelection().anchorNode.parentNode.click()')
|
|
else:
|
|
try:
|
|
selected_element = xml.etree.ElementTree.fromstring(
|
|
'<html>' + widget.selectedHtml() + '</html>').find('a')
|
|
except xml.etree.ElementTree.ParseError:
|
|
raise cmdexc.CommandError('Could not parse selected element!')
|
|
|
|
if selected_element is not None:
|
|
try:
|
|
url = selected_element.attrib['href']
|
|
except KeyError:
|
|
raise cmdexc.CommandError('Anchor element without href!')
|
|
url = self._current_url().resolved(QUrl(url))
|
|
self._open(url, tab)
|
|
|
|
@cmdutils.register(instance='command-dispatcher', name='inspector',
|
|
scope='window')
|
|
def toggle_inspector(self):
|
|
"""Toggle the web inspector.
|
|
|
|
Note: Due a bug in Qt, the inspector will show incorrect request
|
|
headers in the network tab.
|
|
"""
|
|
cur = self._current_widget()
|
|
if cur.inspector is None:
|
|
if not config.get('general', 'developer-extras'):
|
|
raise cmdexc.CommandError(
|
|
"Please enable developer-extras before using the "
|
|
"webinspector!")
|
|
cur.inspector = inspector.WebInspector()
|
|
cur.inspector.setPage(cur.page())
|
|
cur.inspector.show()
|
|
elif cur.inspector.isVisible():
|
|
cur.inspector.hide()
|
|
else:
|
|
if not config.get('general', 'developer-extras'):
|
|
raise cmdexc.CommandError(
|
|
"Please enable developer-extras before using the "
|
|
"webinspector!")
|
|
else:
|
|
cur.inspector.show()
|
|
|
|
@cmdutils.register(instance='command-dispatcher', scope='window')
|
|
def download(self, url=None, dest_old: {'hide': True}=None, *,
|
|
mhtml_=False, dest=None):
|
|
"""Download a given URL, or current page if no URL given.
|
|
|
|
The form `:download [url] [dest]` is deprecated, use `:download --dest
|
|
[dest] [url]` instead.
|
|
|
|
Args:
|
|
url: The URL to download. If not given, download the current page.
|
|
dest_old: (deprecated) Same as dest.
|
|
dest: The file path to write the download to, or None to ask.
|
|
mhtml_: Download the current page and all assets as mhtml file.
|
|
"""
|
|
if dest_old is not None:
|
|
message.warning(
|
|
self._win_id, ":download [url] [dest] is deprecated - use"
|
|
" download --dest [dest] [url]")
|
|
if dest is not None:
|
|
raise cmdexc.CommandError("Can't give two destinations for the"
|
|
" download.")
|
|
dest = dest_old
|
|
|
|
download_manager = objreg.get('download-manager', scope='window',
|
|
window=self._win_id)
|
|
if url:
|
|
if mhtml_:
|
|
raise cmdexc.CommandError("Can only download the current page"
|
|
" as mhtml.")
|
|
url = urlutils.qurl_from_user_input(url)
|
|
urlutils.raise_cmdexc_if_invalid(url)
|
|
download_manager.get(url, filename=dest)
|
|
else:
|
|
if mhtml_:
|
|
self._download_mhtml(dest)
|
|
else:
|
|
page = self._current_widget().page()
|
|
download_manager.get(self._current_url(), page=page,
|
|
filename=dest)
|
|
|
|
def _download_mhtml(self, dest=None):
|
|
"""Download the current page as a MHTML file, including all assets.
|
|
|
|
Args:
|
|
dest: The file path to write the download to.
|
|
"""
|
|
web_view = self._current_widget()
|
|
if dest is None:
|
|
suggested_fn = self._current_title() + ".mht"
|
|
suggested_fn = utils.sanitize_filename(suggested_fn)
|
|
filename, q = downloads.ask_for_filename(
|
|
suggested_fn, self._win_id, parent=web_view,
|
|
)
|
|
if filename is not None:
|
|
mhtml.start_download_checked(filename, web_view=web_view)
|
|
else:
|
|
q.answered.connect(functools.partial(
|
|
mhtml.start_download_checked, web_view=web_view))
|
|
q.ask()
|
|
else:
|
|
mhtml.start_download_checked(dest, web_view=web_view)
|
|
|
|
@cmdutils.register(instance='command-dispatcher', scope='window')
|
|
def view_source(self):
|
|
"""Show the source of the current page."""
|
|
# pylint: disable=no-member
|
|
# WORKAROUND for https://bitbucket.org/logilab/pylint/issue/491/
|
|
widget = self._current_widget()
|
|
if widget.viewing_source:
|
|
raise cmdexc.CommandError("Already viewing source!")
|
|
frame = widget.page().currentFrame()
|
|
html = frame.toHtml()
|
|
lexer = pygments.lexers.HtmlLexer()
|
|
formatter = pygments.formatters.HtmlFormatter(
|
|
full=True, linenos='table')
|
|
highlighted = pygments.highlight(html, lexer, formatter)
|
|
current_url = self._current_url()
|
|
tab = self._tabbed_browser.tabopen(explicit=True)
|
|
tab.setHtml(highlighted, current_url)
|
|
tab.viewing_source = True
|
|
|
|
@cmdutils.register(instance='command-dispatcher', scope='window',
|
|
debug=True)
|
|
def debug_dump_page(self, dest, plain=False):
|
|
"""Dump the current page's content to a file.
|
|
|
|
Args:
|
|
dest: Where to write the file to.
|
|
plain: Write plain text instead of HTML.
|
|
"""
|
|
web_view = self._current_widget()
|
|
mainframe = web_view.page().mainFrame()
|
|
if plain:
|
|
data = mainframe.toPlainText()
|
|
else:
|
|
data = mainframe.toHtml()
|
|
|
|
dest = os.path.expanduser(dest)
|
|
|
|
try:
|
|
with open(dest, 'w', encoding='utf-8') as f:
|
|
f.write(data)
|
|
except OSError as e:
|
|
raise cmdexc.CommandError('Could not write page: {}'.format(e))
|
|
else:
|
|
message.info(self._win_id, "Dumped page to {}.".format(dest))
|
|
|
|
@cmdutils.register(instance='command-dispatcher', name='help',
|
|
completion=[usertypes.Completion.helptopic],
|
|
scope='window')
|
|
def show_help(self, tab=False, bg=False, window=False, topic=None):
|
|
r"""Show help about a command or setting.
|
|
|
|
Args:
|
|
tab: Open in a new tab.
|
|
bg: Open in a background tab.
|
|
window: Open in a new window.
|
|
topic: The topic to show help for.
|
|
|
|
- :__command__ for commands.
|
|
- __section__\->__option__ for settings.
|
|
"""
|
|
if topic is None:
|
|
path = 'index.html'
|
|
elif topic.startswith(':'):
|
|
command = topic[1:]
|
|
if command not in cmdutils.cmd_dict:
|
|
raise cmdexc.CommandError("Invalid command {}!".format(
|
|
command))
|
|
path = 'commands.html#{}'.format(command)
|
|
elif '->' in topic:
|
|
parts = topic.split('->')
|
|
if len(parts) != 2:
|
|
raise cmdexc.CommandError("Invalid help topic {}!".format(
|
|
topic))
|
|
try:
|
|
config.get(*parts)
|
|
except configexc.NoSectionError:
|
|
raise cmdexc.CommandError("Invalid section {}!".format(
|
|
parts[0]))
|
|
except configexc.NoOptionError:
|
|
raise cmdexc.CommandError("Invalid option {}!".format(
|
|
parts[1]))
|
|
path = 'settings.html#{}'.format(topic.replace('->', '-'))
|
|
else:
|
|
raise cmdexc.CommandError("Invalid help topic {}!".format(topic))
|
|
url = QUrl('qute://help/{}'.format(path))
|
|
self._open(url, tab, bg, window)
|
|
|
|
@cmdutils.register(instance='command-dispatcher',
|
|
modes=[KeyMode.insert], hide=True, scope='window')
|
|
def open_editor(self):
|
|
"""Open an external editor with the currently selected form field.
|
|
|
|
The editor which should be launched can be configured via the
|
|
`general -> editor` config option.
|
|
"""
|
|
frame = self._current_widget().page().currentFrame()
|
|
try:
|
|
elem = webelem.focus_elem(frame)
|
|
except webelem.IsNullError:
|
|
raise cmdexc.CommandError("No element focused!")
|
|
if not elem.is_editable(strict=True):
|
|
raise cmdexc.CommandError("Focused element is not editable!")
|
|
if elem.is_content_editable():
|
|
text = str(elem)
|
|
else:
|
|
text = elem.evaluateJavaScript('this.value')
|
|
self._editor = editor.ExternalEditor(
|
|
self._win_id, self._tabbed_browser)
|
|
self._editor.editing_finished.connect(
|
|
functools.partial(self.on_editing_finished, elem))
|
|
self._editor.edit(text)
|
|
|
|
def on_editing_finished(self, elem, text):
|
|
"""Write the editor text into the form field and clean up tempfile.
|
|
|
|
Callback for GUIProcess when the editor was closed.
|
|
|
|
Args:
|
|
elem: The WebElementWrapper which was modified.
|
|
text: The new text to insert.
|
|
"""
|
|
try:
|
|
if elem.is_content_editable():
|
|
log.misc.debug("Filling element {} via setPlainText.".format(
|
|
elem.debug_text()))
|
|
elem.setPlainText(text)
|
|
else:
|
|
log.misc.debug("Filling element {} via javascript.".format(
|
|
elem.debug_text()))
|
|
text = webelem.javascript_escape(text)
|
|
elem.evaluateJavaScript("this.value='{}'".format(text))
|
|
except webelem.IsNullError:
|
|
raise cmdexc.CommandError("Element vanished while editing!")
|
|
|
|
def _clear_search(self, view, text):
|
|
"""Clear search string/highlights for the given view.
|
|
|
|
This does nothing if the view's search text is the same as the given
|
|
text.
|
|
"""
|
|
if view.search_text is not None and view.search_text != text:
|
|
# We first clear the marked text, then the highlights
|
|
view.search('', 0)
|
|
view.search('', QWebPage.HighlightAllOccurrences)
|
|
|
|
@cmdutils.register(instance='command-dispatcher', scope='window',
|
|
maxsplit=0)
|
|
def search(self, text="", reverse=False):
|
|
"""Search for a text on the current page. With no text, clear results.
|
|
|
|
Args:
|
|
text: The text to search for.
|
|
reverse: Reverse search direction.
|
|
"""
|
|
view = self._current_widget()
|
|
self._clear_search(view, text)
|
|
flags = 0
|
|
ignore_case = config.get('general', 'ignore-case')
|
|
if ignore_case == 'smart':
|
|
if not text.islower():
|
|
flags |= QWebPage.FindCaseSensitively
|
|
elif not ignore_case:
|
|
flags |= QWebPage.FindCaseSensitively
|
|
if config.get('general', 'wrap-search'):
|
|
flags |= QWebPage.FindWrapsAroundDocument
|
|
if reverse:
|
|
flags |= QWebPage.FindBackward
|
|
# We actually search *twice* - once to highlight everything, then again
|
|
# to get a mark so we can navigate.
|
|
view.search(text, flags)
|
|
view.search(text, flags | QWebPage.HighlightAllOccurrences)
|
|
view.search_text = text
|
|
view.search_flags = flags
|
|
self._tabbed_browser.search_text = text
|
|
self._tabbed_browser.search_flags = flags
|
|
|
|
@cmdutils.register(instance='command-dispatcher', hide=True,
|
|
scope='window', count='count')
|
|
def search_next(self, count=1):
|
|
"""Continue the search to the ([count]th) next term.
|
|
|
|
Args:
|
|
count: How many elements to ignore.
|
|
"""
|
|
view = self._current_widget()
|
|
|
|
self._clear_search(view, self._tabbed_browser.search_text)
|
|
|
|
if self._tabbed_browser.search_text is not None:
|
|
view.search_text = self._tabbed_browser.search_text
|
|
view.search_flags = self._tabbed_browser.search_flags
|
|
view.search(view.search_text,
|
|
view.search_flags | QWebPage.HighlightAllOccurrences)
|
|
for _ in range(count):
|
|
view.search(view.search_text, view.search_flags)
|
|
|
|
@cmdutils.register(instance='command-dispatcher', hide=True,
|
|
scope='window', count='count')
|
|
def search_prev(self, count=1):
|
|
"""Continue the search to the ([count]th) previous term.
|
|
|
|
Args:
|
|
count: How many elements to ignore.
|
|
"""
|
|
view = self._current_widget()
|
|
self._clear_search(view, self._tabbed_browser.search_text)
|
|
|
|
if self._tabbed_browser.search_text is not None:
|
|
view.search_text = self._tabbed_browser.search_text
|
|
view.search_flags = self._tabbed_browser.search_flags
|
|
view.search(view.search_text,
|
|
view.search_flags | QWebPage.HighlightAllOccurrences)
|
|
# The int() here serves as a QFlags constructor to create a copy of the
|
|
# QFlags instance rather as a reference. I don't know why it works this
|
|
# way, but it does.
|
|
flags = int(view.search_flags)
|
|
if flags & QWebPage.FindBackward:
|
|
flags &= ~QWebPage.FindBackward
|
|
else:
|
|
flags |= QWebPage.FindBackward
|
|
for _ in range(count):
|
|
view.search(view.search_text, flags)
|
|
|
|
@cmdutils.register(instance='command-dispatcher', hide=True,
|
|
modes=[KeyMode.caret], scope='window', count='count')
|
|
def move_to_next_line(self, count=1):
|
|
"""Move the cursor or selection to the next line.
|
|
|
|
Args:
|
|
count: How many lines to move.
|
|
"""
|
|
webview = self._current_widget()
|
|
if not webview.selection_enabled:
|
|
act = QWebPage.MoveToNextLine
|
|
else:
|
|
act = QWebPage.SelectNextLine
|
|
for _ in range(count):
|
|
webview.triggerPageAction(act)
|
|
|
|
@cmdutils.register(instance='command-dispatcher', hide=True,
|
|
modes=[KeyMode.caret], scope='window', count='count')
|
|
def move_to_prev_line(self, count=1):
|
|
"""Move the cursor or selection to the prev line.
|
|
|
|
Args:
|
|
count: How many lines to move.
|
|
"""
|
|
webview = self._current_widget()
|
|
if not webview.selection_enabled:
|
|
act = QWebPage.MoveToPreviousLine
|
|
else:
|
|
act = QWebPage.SelectPreviousLine
|
|
for _ in range(count):
|
|
webview.triggerPageAction(act)
|
|
|
|
@cmdutils.register(instance='command-dispatcher', hide=True,
|
|
modes=[KeyMode.caret], scope='window', count='count')
|
|
def move_to_next_char(self, count=1):
|
|
"""Move the cursor or selection to the next char.
|
|
|
|
Args:
|
|
count: How many lines to move.
|
|
"""
|
|
webview = self._current_widget()
|
|
if not webview.selection_enabled:
|
|
act = QWebPage.MoveToNextChar
|
|
else:
|
|
act = QWebPage.SelectNextChar
|
|
for _ in range(count):
|
|
webview.triggerPageAction(act)
|
|
|
|
@cmdutils.register(instance='command-dispatcher', hide=True,
|
|
modes=[KeyMode.caret], scope='window', count='count')
|
|
def move_to_prev_char(self, count=1):
|
|
"""Move the cursor or selection to the previous char.
|
|
|
|
Args:
|
|
count: How many chars to move.
|
|
"""
|
|
webview = self._current_widget()
|
|
if not webview.selection_enabled:
|
|
act = QWebPage.MoveToPreviousChar
|
|
else:
|
|
act = QWebPage.SelectPreviousChar
|
|
for _ in range(count):
|
|
webview.triggerPageAction(act)
|
|
|
|
@cmdutils.register(instance='command-dispatcher', hide=True,
|
|
modes=[KeyMode.caret], scope='window', count='count')
|
|
def move_to_end_of_word(self, count=1):
|
|
"""Move the cursor or selection to the end of the word.
|
|
|
|
Args:
|
|
count: How many words to move.
|
|
"""
|
|
webview = self._current_widget()
|
|
if not webview.selection_enabled:
|
|
act = [QWebPage.MoveToNextWord]
|
|
if sys.platform == 'win32':
|
|
act.append(QWebPage.MoveToPreviousChar)
|
|
else:
|
|
act = [QWebPage.SelectNextWord]
|
|
if sys.platform == 'win32':
|
|
act.append(QWebPage.SelectPreviousChar)
|
|
for _ in range(count):
|
|
for a in act:
|
|
webview.triggerPageAction(a)
|
|
|
|
@cmdutils.register(instance='command-dispatcher', hide=True,
|
|
modes=[KeyMode.caret], scope='window', count='count')
|
|
def move_to_next_word(self, count=1):
|
|
"""Move the cursor or selection to the next word.
|
|
|
|
Args:
|
|
count: How many words to move.
|
|
"""
|
|
webview = self._current_widget()
|
|
if not webview.selection_enabled:
|
|
act = [QWebPage.MoveToNextWord]
|
|
if sys.platform != 'win32':
|
|
act.append(QWebPage.MoveToNextChar)
|
|
else:
|
|
act = [QWebPage.SelectNextWord]
|
|
if sys.platform != 'win32':
|
|
act.append(QWebPage.SelectNextChar)
|
|
for _ in range(count):
|
|
for a in act:
|
|
webview.triggerPageAction(a)
|
|
|
|
@cmdutils.register(instance='command-dispatcher', hide=True,
|
|
modes=[KeyMode.caret], scope='window', count='count')
|
|
def move_to_prev_word(self, count=1):
|
|
"""Move the cursor or selection to the previous word.
|
|
|
|
Args:
|
|
count: How many words to move.
|
|
"""
|
|
webview = self._current_widget()
|
|
if not webview.selection_enabled:
|
|
act = QWebPage.MoveToPreviousWord
|
|
else:
|
|
act = QWebPage.SelectPreviousWord
|
|
for _ in range(count):
|
|
webview.triggerPageAction(act)
|
|
|
|
@cmdutils.register(instance='command-dispatcher', hide=True,
|
|
modes=[KeyMode.caret], scope='window')
|
|
def move_to_start_of_line(self):
|
|
"""Move the cursor or selection to the start of the line."""
|
|
webview = self._current_widget()
|
|
if not webview.selection_enabled:
|
|
act = QWebPage.MoveToStartOfLine
|
|
else:
|
|
act = QWebPage.SelectStartOfLine
|
|
webview.triggerPageAction(act)
|
|
|
|
@cmdutils.register(instance='command-dispatcher', hide=True,
|
|
modes=[KeyMode.caret], scope='window')
|
|
def move_to_end_of_line(self):
|
|
"""Move the cursor or selection to the end of line."""
|
|
webview = self._current_widget()
|
|
if not webview.selection_enabled:
|
|
act = QWebPage.MoveToEndOfLine
|
|
else:
|
|
act = QWebPage.SelectEndOfLine
|
|
webview.triggerPageAction(act)
|
|
|
|
@cmdutils.register(instance='command-dispatcher', hide=True,
|
|
modes=[KeyMode.caret], scope='window', count='count')
|
|
def move_to_start_of_next_block(self, count=1):
|
|
"""Move the cursor or selection to the start of next block.
|
|
|
|
Args:
|
|
count: How many blocks to move.
|
|
"""
|
|
webview = self._current_widget()
|
|
if not webview.selection_enabled:
|
|
act = [QWebPage.MoveToNextLine,
|
|
QWebPage.MoveToStartOfBlock]
|
|
else:
|
|
act = [QWebPage.SelectNextLine,
|
|
QWebPage.SelectStartOfBlock]
|
|
for _ in range(count):
|
|
for a in act:
|
|
webview.triggerPageAction(a)
|
|
|
|
@cmdutils.register(instance='command-dispatcher', hide=True,
|
|
modes=[KeyMode.caret], scope='window', count='count')
|
|
def move_to_start_of_prev_block(self, count=1):
|
|
"""Move the cursor or selection to the start of previous block.
|
|
|
|
Args:
|
|
count: How many blocks to move.
|
|
"""
|
|
webview = self._current_widget()
|
|
if not webview.selection_enabled:
|
|
act = [QWebPage.MoveToPreviousLine,
|
|
QWebPage.MoveToStartOfBlock]
|
|
else:
|
|
act = [QWebPage.SelectPreviousLine,
|
|
QWebPage.SelectStartOfBlock]
|
|
for _ in range(count):
|
|
for a in act:
|
|
webview.triggerPageAction(a)
|
|
|
|
@cmdutils.register(instance='command-dispatcher', hide=True,
|
|
modes=[KeyMode.caret], scope='window', count='count')
|
|
def move_to_end_of_next_block(self, count=1):
|
|
"""Move the cursor or selection to the end of next block.
|
|
|
|
Args:
|
|
count: How many blocks to move.
|
|
"""
|
|
webview = self._current_widget()
|
|
if not webview.selection_enabled:
|
|
act = [QWebPage.MoveToNextLine,
|
|
QWebPage.MoveToEndOfBlock]
|
|
else:
|
|
act = [QWebPage.SelectNextLine,
|
|
QWebPage.SelectEndOfBlock]
|
|
for _ in range(count):
|
|
for a in act:
|
|
webview.triggerPageAction(a)
|
|
|
|
@cmdutils.register(instance='command-dispatcher', hide=True,
|
|
modes=[KeyMode.caret], scope='window', count='count')
|
|
def move_to_end_of_prev_block(self, count=1):
|
|
"""Move the cursor or selection to the end of previous block.
|
|
|
|
Args:
|
|
count: How many blocks to move.
|
|
"""
|
|
webview = self._current_widget()
|
|
if not webview.selection_enabled:
|
|
act = [QWebPage.MoveToPreviousLine, QWebPage.MoveToEndOfBlock]
|
|
else:
|
|
act = [QWebPage.SelectPreviousLine, QWebPage.SelectEndOfBlock]
|
|
for _ in range(count):
|
|
for a in act:
|
|
webview.triggerPageAction(a)
|
|
|
|
@cmdutils.register(instance='command-dispatcher', hide=True,
|
|
modes=[KeyMode.caret], scope='window')
|
|
def move_to_start_of_document(self):
|
|
"""Move the cursor or selection to the start of the document."""
|
|
webview = self._current_widget()
|
|
if not webview.selection_enabled:
|
|
act = QWebPage.MoveToStartOfDocument
|
|
else:
|
|
act = QWebPage.SelectStartOfDocument
|
|
webview.triggerPageAction(act)
|
|
|
|
@cmdutils.register(instance='command-dispatcher', hide=True,
|
|
modes=[KeyMode.caret], scope='window')
|
|
def move_to_end_of_document(self):
|
|
"""Move the cursor or selection to the end of the document."""
|
|
webview = self._current_widget()
|
|
if not webview.selection_enabled:
|
|
act = QWebPage.MoveToEndOfDocument
|
|
else:
|
|
act = QWebPage.SelectEndOfDocument
|
|
webview.triggerPageAction(act)
|
|
|
|
@cmdutils.register(instance='command-dispatcher', scope='window')
|
|
def yank_selected(self, sel=False, keep=False):
|
|
"""Yank the selected text to the clipboard or primary selection.
|
|
|
|
Args:
|
|
sel: Use the primary selection instead of the clipboard.
|
|
keep: If given, stay in visual mode after yanking.
|
|
"""
|
|
s = self._current_widget().selectedText()
|
|
if not self._current_widget().hasSelection() or len(s) == 0:
|
|
message.info(self._win_id, "Nothing to yank")
|
|
return
|
|
|
|
clipboard = QApplication.clipboard()
|
|
if sel and clipboard.supportsSelection():
|
|
mode = QClipboard.Selection
|
|
target = "primary selection"
|
|
else:
|
|
mode = QClipboard.Clipboard
|
|
target = "clipboard"
|
|
clipboard.setText(s, mode)
|
|
message.info(self._win_id, "{} {} yanked to {}".format(
|
|
len(s), "char" if len(s) == 1 else "chars", target))
|
|
if not keep:
|
|
modeman.maybe_leave(self._win_id, KeyMode.caret, "yank selected")
|
|
|
|
@cmdutils.register(instance='command-dispatcher', hide=True,
|
|
modes=[KeyMode.caret], scope='window')
|
|
def toggle_selection(self):
|
|
"""Toggle caret selection mode."""
|
|
widget = self._current_widget()
|
|
widget.selection_enabled = not widget.selection_enabled
|
|
mainwindow = objreg.get('main-window', scope='window',
|
|
window=self._win_id)
|
|
mainwindow.status.set_mode_active(usertypes.KeyMode.caret, True)
|
|
|
|
@cmdutils.register(instance='command-dispatcher', hide=True,
|
|
modes=[KeyMode.caret], scope='window')
|
|
def drop_selection(self):
|
|
"""Drop selection and keep selection mode enabled."""
|
|
self._current_widget().triggerPageAction(QWebPage.MoveToNextChar)
|
|
|
|
@cmdutils.register(instance='command-dispatcher', scope='window',
|
|
count='count', debug=True)
|
|
def debug_webaction(self, action, count=1):
|
|
"""Execute a webaction.
|
|
|
|
See http://doc.qt.io/qt-5/qwebpage.html#WebAction-enum for the
|
|
available actions.
|
|
|
|
Args:
|
|
action: The action to execute, e.g. MoveToNextChar.
|
|
count: How many times to repeat the action.
|
|
"""
|
|
member = getattr(QWebPage, action, None)
|
|
if not isinstance(member, QWebPage.WebAction):
|
|
raise cmdexc.CommandError("{} is not a valid web action!".format(
|
|
action))
|
|
view = self._current_widget()
|
|
for _ in range(count):
|
|
view.triggerPageAction(member)
|
|
|
|
@cmdutils.register(instance='command-dispatcher', scope='window',
|
|
maxsplit=0, no_cmd_split=True)
|
|
def jseval(self, js_code, quiet=False):
|
|
"""Evaluate a JavaScript string.
|
|
|
|
Args:
|
|
js_code: The string to evaluate.
|
|
quiet: Don't show resulting JS object.
|
|
"""
|
|
frame = self._current_widget().page().mainFrame()
|
|
out = frame.evaluateJavaScript(js_code)
|
|
|
|
if quiet:
|
|
return
|
|
|
|
if out is None:
|
|
# Getting the actual error (if any) seems to be difficult. The
|
|
# error does end up in BrowserPage.javaScriptConsoleMessage(), but
|
|
# distinguishing between :jseval errors and errors from the webpage
|
|
# is not trivial...
|
|
message.info(self._win_id, 'No output or error')
|
|
else:
|
|
# The output can be a string, number, dict, array, etc. But *don't*
|
|
# output too much data, as this will make qutebrowser hang
|
|
out = str(out)
|
|
if len(out) > 5000:
|
|
message.info(self._win_id, out[:5000] + ' [...trimmed...]')
|
|
else:
|
|
message.info(self._win_id, out)
|
|
|
|
@cmdutils.register(instance='command-dispatcher', scope='window')
|
|
def fake_key(self, keystring, global_=False):
|
|
"""Send a fake keypress or key string to the website or qutebrowser.
|
|
|
|
:fake-key xy - sends the keychain 'xy'
|
|
:fake-key <Ctrl-x> - sends Ctrl-x
|
|
:fake-key <Escape> - sends the escape key
|
|
|
|
Args:
|
|
keystring: The keystring to send.
|
|
global_: If given, the keys are sent to the qutebrowser UI.
|
|
"""
|
|
try:
|
|
keyinfos = utils.parse_keystring(keystring)
|
|
except utils.KeyParseError as e:
|
|
raise cmdexc.CommandError(str(e))
|
|
|
|
for keyinfo in keyinfos:
|
|
press_event = QKeyEvent(QEvent.KeyPress, keyinfo.key,
|
|
keyinfo.modifiers, keyinfo.text)
|
|
release_event = QKeyEvent(QEvent.KeyRelease, keyinfo.key,
|
|
keyinfo.modifiers, keyinfo.text)
|
|
|
|
if global_:
|
|
receiver = QApplication.focusWindow()
|
|
if receiver is None:
|
|
raise cmdexc.CommandError("No focused window!")
|
|
else:
|
|
try:
|
|
receiver = objreg.get('webview', scope='tab',
|
|
tab='current')
|
|
except objreg.RegistryUnavailableError:
|
|
raise cmdexc.CommandError("No focused webview!")
|
|
|
|
QApplication.postEvent(receiver, press_event)
|
|
QApplication.postEvent(receiver, release_event)
|