qutebrowser/qutebrowser/browser/curcommand.py

411 lines
14 KiB
Python
Raw Normal View History

2014-04-17 09:44:26 +02:00
# Copyright 2014 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 main tabbed browser widget."""
2014-04-29 18:00:22 +02:00
import os
import logging
from tempfile import mkstemp
from functools import partial
2014-04-22 12:10:27 +02:00
from PyQt5.QtWidgets import QApplication
2014-04-29 18:00:22 +02:00
from PyQt5.QtCore import pyqtSlot, Qt, QObject, QProcess
2014-04-17 09:44:26 +02:00
from PyQt5.QtGui import QClipboard
2014-04-22 11:16:45 +02:00
from PyQt5.QtPrintSupport import QPrinter, QPrintDialog, QPrintPreviewDialog
2014-04-17 09:44:26 +02:00
import qutebrowser.utils.url as urlutils
2014-04-22 10:08:56 +02:00
import qutebrowser.utils.message as message
2014-04-17 09:44:26 +02:00
import qutebrowser.commands.utils as cmdutils
2014-04-29 18:00:22 +02:00
import qutebrowser.utils.webelem as webelem
import qutebrowser.config.config as config
2014-04-17 09:44:26 +02:00
class CurCommandDispatcher(QObject):
"""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.cur because at the time
cmdutils.register() decorators are run, currentWidget() will return None.
Attributes:
2014-04-17 17:44:27 +02:00
_tabs: The TabbedBrowser object.
2014-04-17 09:44:26 +02:00
"""
def __init__(self, parent):
"""Constructor.
Args:
parent: The TabbedBrowser for this dispatcher.
"""
super().__init__(parent)
2014-04-17 17:44:27 +02:00
self._tabs = parent
2014-04-17 09:44:26 +02:00
def _scroll_percent(self, perc=None, count=None, orientation=None):
"""Inner logic for scroll_percent_(x|y).
Args:
perc: How many percent to scroll, or None
count: How many percent to scroll, or None
orientation: Qt.Horizontal or Qt.Vertical
"""
if perc is None and count is None:
perc = 100
elif perc is None:
perc = int(count)
else:
perc = float(perc)
frame = self._tabs.currentWidget().page_.currentFrame()
2014-04-17 09:44:26 +02:00
m = frame.scrollBarMaximum(orientation)
if m == 0:
return
frame.setScrollBarValue(orientation, int(m * perc / 100))
@cmdutils.register(instance='mainwindow.tabs.cur', name='open', maxsplit=0)
2014-04-17 09:44:26 +02:00
def openurl(self, url, count=None):
"""Open an url in the current/[count]th tab.
Command handler for :open.
Args:
url: The URL to open.
count: The tab index to open the URL in, or None.
"""
2014-04-17 17:44:27 +02:00
tab = self._tabs.cntwidget(count)
2014-04-17 09:44:26 +02:00
if tab is None:
if count is None:
# We want to open an URL in the current tab, but none exists
# yet.
2014-04-17 17:44:27 +02:00
self._tabs.tabopen(url)
2014-04-17 09:44:26 +02:00
else:
# Explicit count with a tab that doesn't exist.
return
else:
tab.openurl(url)
@cmdutils.register(instance='mainwindow.tabs.cur', name='reload')
def reloadpage(self, count=None):
"""Reload the current/[count]th tab.
Command handler for :reload.
Args:
count: The tab index to reload, or None.
"""
2014-04-17 17:44:27 +02:00
tab = self._tabs.cntwidget(count)
2014-04-17 09:44:26 +02:00
if tab is not None:
tab.reload()
@cmdutils.register(instance='mainwindow.tabs.cur')
def stop(self, count=None):
"""Stop loading in the current/[count]th tab.
Command handler for :stop.
Args:
count: The tab index to stop, or None.
"""
2014-04-17 17:44:27 +02:00
tab = self._tabs.cntwidget(count)
2014-04-17 09:44:26 +02:00
if tab is not None:
tab.stop()
2014-04-22 11:16:45 +02:00
@cmdutils.register(instance='mainwindow.tabs.cur', name='printpreview')
def printpreview(self, count=None):
"""Preview printing of the current/[count]th tab.
Command handler for :printpreview.
Args:
count: The tab index to print, or None.
"""
tab = self._tabs.cntwidget(count)
if tab is not None:
preview = QPrintPreviewDialog(tab)
preview.paintRequested.connect(tab.print)
preview.exec_()
2014-04-17 09:44:26 +02:00
@cmdutils.register(instance='mainwindow.tabs.cur', name='print')
def printpage(self, count=None):
"""Print the current/[count]th tab.
Command handler for :print.
Args:
count: The tab index to print, or None.
"""
2014-04-22 14:26:07 +02:00
# QTBUG: We only get blank pages.
2014-04-22 11:16:45 +02:00
# https://bugreports.qt-project.org/browse/QTBUG-19571
2014-04-22 14:26:07 +02:00
# If this isn't fixed in Qt 5.3, bug should be reopened.
2014-04-17 17:44:27 +02:00
tab = self._tabs.cntwidget(count)
2014-04-17 09:44:26 +02:00
if tab is not None:
2014-04-22 11:16:45 +02:00
printer = QPrinter()
printdiag = QPrintDialog(printer, tab)
printdiag.open(lambda: tab.print(printdiag.printer()))
2014-04-17 09:44:26 +02:00
@cmdutils.register(instance='mainwindow.tabs.cur')
def back(self, count=1):
"""Go back in the history of the current tab.
Command handler for :back.
Args:
count: How many pages to go back.
"""
for _ in range(count):
if self._tabs.currentWidget().page_.history().canGoBack():
self._tabs.currentWidget().back()
else:
2014-04-23 06:17:29 +02:00
message.error("At beginning of history.")
break
2014-04-17 09:44:26 +02:00
@cmdutils.register(instance='mainwindow.tabs.cur')
def forward(self, count=1):
"""Go forward in the history of the current tab.
Command handler for :forward.
Args:
count: How many pages to go forward.
"""
for _ in range(count):
if self._tabs.currentWidget().page_.history().canGoForward():
self._tabs.currentWidget().forward()
else:
2014-04-23 06:17:29 +02:00
message.error("At end of history.")
break
2014-04-17 09:44:26 +02:00
2014-04-19 17:50:11 +02:00
@cmdutils.register(instance='mainwindow.tabs.cur')
2014-04-25 16:53:23 +02:00
def hint(self, mode='all', target='normal'):
2014-04-19 17:50:11 +02:00
"""Start hinting.
Command handler for :hint.
Args:
mode: The hinting mode to use.
2014-04-21 19:29:11 +02:00
target: Where to open the links.
2014-04-19 17:50:11 +02:00
"""
widget = self._tabs.currentWidget()
frame = widget.page_.currentFrame()
if frame is None:
message.error("No frame focused!")
else:
widget.hintmanager.start(frame, widget.url(), mode, target)
2014-04-19 17:50:11 +02:00
2014-04-27 21:59:23 +02:00
@cmdutils.register(instance='mainwindow.tabs.cur', hide=True)
def follow_hint(self):
"""Follow the currently selected hint."""
self._tabs.currentWidget().hintmanager.follow_hint()
2014-04-21 15:20:41 +02:00
@pyqtSlot(str)
def handle_hint_key(self, keystr):
"""Handle a new hint keypress."""
self._tabs.currentWidget().hintmanager.handle_partial_key(keystr)
@pyqtSlot(str)
def fire_hint(self, keystr):
"""Fire a completed hint."""
self._tabs.currentWidget().hintmanager.fire(keystr)
2014-04-17 09:44:26 +02:00
@pyqtSlot(str, int)
def search(self, text, flags):
"""Search for text in the current page.
Args:
text: The text to search for.
flags: The QWebPage::FindFlags.
"""
2014-04-17 17:44:27 +02:00
self._tabs.currentWidget().findText(text, flags)
2014-04-17 09:44:26 +02:00
@cmdutils.register(instance='mainwindow.tabs.cur', hide=True)
def scroll(self, dx, dy, count=1):
"""Scroll the current tab by count * dx/dy.
Command handler for :scroll.
Args:
dx: How much to scroll in x-direction.
dy: How much to scroll in x-direction.
count: multiplier
"""
dx = int(count) * float(dx)
dy = int(count) * float(dy)
self._tabs.currentWidget().page_.currentFrame().scroll(dx, dy)
2014-04-17 09:44:26 +02:00
@cmdutils.register(instance='mainwindow.tabs.cur', name='scroll_perc_x',
hide=True)
def scroll_percent_x(self, perc=None, count=None):
"""Scroll the current tab to a specific percent of the page (horiz).
Command handler for :scroll_perc_x.
Args:
perc: Percentage to scroll.
count: Percentage to scroll.
"""
self._scroll_percent(perc, count, Qt.Horizontal)
@cmdutils.register(instance='mainwindow.tabs.cur', name='scroll_perc_y',
hide=True)
def scroll_percent_y(self, perc=None, count=None):
"""Scroll the current tab to a specific percent of the page (vert).
Command handler for :scroll_perc_y
Args:
perc: Percentage to scroll.
count: Percentage to scroll.
"""
self._scroll_percent(perc, count, Qt.Vertical)
@cmdutils.register(instance='mainwindow.tabs.cur', hide=True)
def scroll_page(self, mx, my, count=1):
"""Scroll the frame page-wise.
Args:
mx: How many pages to scroll to the right.
my: How many pages to scroll down.
count: multiplier
"""
frame = self._tabs.currentWidget().page_.currentFrame()
size = frame.geometry()
frame.scroll(int(count) * float(mx) * size.width(),
int(count) * float(my) * size.height())
2014-04-17 09:44:26 +02:00
@cmdutils.register(instance='mainwindow.tabs.cur')
def yank(self, sel=False):
"""Yank the current url to the clipboard or primary selection.
Command handler for :yank.
Args:
sel: True to use primary selection, False to use clipboard
"""
clip = QApplication.clipboard()
2014-04-17 17:44:27 +02:00
url = urlutils.urlstring(self._tabs.currentWidget().url())
2014-04-17 09:44:26 +02:00
mode = QClipboard.Selection if sel else QClipboard.Clipboard
clip.setText(url, mode)
2014-04-25 16:53:23 +02:00
message.info("URL yanked to {}".format("primary selection" if sel
else "clipboard"))
2014-04-17 09:44:26 +02:00
@cmdutils.register(instance='mainwindow.tabs.cur', name='yanktitle')
def yank_title(self, sel=False):
"""Yank the current title to the clipboard or primary selection.
Command handler for :yanktitle.
Args:
sel: True to use primary selection, False to use clipboard
"""
clip = QApplication.clipboard()
2014-04-17 17:44:27 +02:00
title = self._tabs.tabText(self._tabs.currentIndex())
2014-04-17 09:44:26 +02:00
mode = QClipboard.Selection if sel else QClipboard.Clipboard
clip.setText(title, mode)
2014-04-25 16:53:23 +02:00
message.info("Title yanked to {}".format("primary selection" if sel
else "clipboard"))
2014-04-17 09:44:26 +02:00
@cmdutils.register(instance='mainwindow.tabs.cur', name='zoomin')
def zoom_in(self, count=1):
"""Zoom in in the current tab.
Args:
count: How many steps to take.
"""
2014-04-17 17:44:27 +02:00
tab = self._tabs.currentWidget()
2014-04-17 09:44:26 +02:00
tab.zoom(count)
@cmdutils.register(instance='mainwindow.tabs.cur', name='zoomout')
def zoom_out(self, count=1):
"""Zoom out in the current tab.
Args:
count: How many steps to take.
"""
2014-04-17 17:44:27 +02:00
tab = self._tabs.currentWidget()
2014-04-17 09:44:26 +02:00
tab.zoom(-count)
2014-04-29 18:00:22 +02:00
@cmdutils.register(instance='mainwindow.tabs.cur', modes=['insert'],
2014-04-29 23:06:07 +02:00
name='open_editor', hide=True, needs_js=True)
2014-04-29 18:00:22 +02:00
def editor(self):
"""Open an external editor with the current form field."""
frame = self._tabs.currentWidget().page_.currentFrame()
elem = frame.findFirstElement(webelem.SELECTORS['editable_focused'])
if elem.isNull():
message.error("No editable element focused!")
return
oshandle, filename = mkstemp(text=True)
text = elem.evaluateJavaScript('this.value')
if text:
with open(filename, 'w') as f:
f.write(text)
proc = QProcess(self)
proc.finished.connect(partial(self.on_editor_closed, elem, oshandle,
filename))
2014-04-30 10:59:43 +02:00
proc.error.connect(partial(self.on_editor_error, oshandle, filename))
2014-04-29 18:00:22 +02:00
editor = config.get('general', 'editor')
executable = editor[0]
args = [arg.replace('{}', filename) for arg in editor[1:]]
logging.debug("Calling \"{}\" with args {}".format(executable, args))
proc.start(executable, args)
2014-04-30 10:59:43 +02:00
def _on_editor_cleanup(self, oshandle, filename):
os.close(oshandle)
try:
os.remove(filename)
except PermissionError:
message.error("Failed to delete tempfile...")
2014-04-29 18:00:22 +02:00
def on_editor_closed(self, elem, oshandle, filename, exitcode,
exitstatus):
2014-04-30 10:46:20 +02:00
"""Write the editor text into the form field and clean up tempfile.
2014-04-29 18:00:22 +02:00
2014-04-30 10:46:20 +02:00
Callback for QProcess when the editor was closed.
2014-04-29 18:00:22 +02:00
"""
logging.debug("Editor closed")
2014-04-30 10:59:43 +02:00
if exitcode != 0:
2014-04-29 18:00:22 +02:00
message.error("Editor did quit abnormally (status {})!".format(
exitcode))
return
2014-04-30 10:59:43 +02:00
if exitstatus != QProcess.NormalExit:
# No error here, since we already handle this in on_editor_error
return
2014-04-29 18:00:22 +02:00
if elem.isNull():
message.error("Element vanished while editing!")
return
with open(filename, 'r') as f:
text = ''.join(f.readlines())
text = webelem.javascript_escape(text)
logging.debug("Read back: {}".format(text))
elem.evaluateJavaScript("this.value='{}'".format(text))
2014-04-30 10:59:43 +02:00
self._on_editor_cleanup(oshandle, filename)
def on_editor_error(self, oshandle, filename, error):
"""Display an error message and clean up when editor crashed."""
messages = {
QProcess.FailedToStart: "The process failed to start.",
QProcess.Crashed: "The process crashed.",
QProcess.Timedout: "The last waitFor...() function timed out.",
QProcess.WriteError: ("An error occurred when attempting to write "
"to the process."),
QProcess.ReadError: ("An error occurred when attempting to read "
"from the process."),
QProcess.UnknownError: "An unknown error occurred.",
}
message.error("Error while calling editor: {}".format(messages[error]))
self._on_editor_cleanup(oshandle, filename)