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)
|
2014-04-22 14:17:17 +02:00
|
|
|
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))
|
|
|
|
|
2014-04-17 12:06:27 +02:00
|
|
|
@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):
|
2014-04-22 14:06:32 +02:00
|
|
|
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.")
|
2014-04-22 14:06:32 +02:00
|
|
|
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):
|
2014-04-22 14:06:32 +02:00
|
|
|
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.")
|
2014-04-22 14:06:32 +02:00
|
|
|
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
|
|
|
"""
|
2014-04-21 23:44:45 +02:00
|
|
|
widget = self._tabs.currentWidget()
|
2014-04-25 13:21:14 +02:00
|
|
|
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)
|
2014-04-22 14:17:17 +02:00
|
|
|
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
|
|
|
|
"""
|
2014-04-22 14:17:17 +02:00
|
|
|
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'],
|
|
|
|
name='open_editor', hide=True)
|
|
|
|
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))
|
|
|
|
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)
|
|
|
|
|
|
|
|
def on_editor_closed(self, elem, oshandle, filename, exitcode,
|
|
|
|
exitstatus):
|
|
|
|
"""Gets called by QProcess when the editor was closed.
|
|
|
|
|
|
|
|
Writes the editor text into the form field.
|
|
|
|
"""
|
|
|
|
logging.debug("Editor closed")
|
|
|
|
if exitcode != 0 or exitstatus != QProcess.NormalExit:
|
|
|
|
message.error("Editor did quit abnormally (status {})!".format(
|
|
|
|
exitcode))
|
|
|
|
return
|
|
|
|
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))
|
|
|
|
os.close(oshandle)
|
|
|
|
try:
|
|
|
|
os.remove(filename)
|
|
|
|
except PermissionError:
|
|
|
|
message.error("Failed to delete tempfile...")
|