qutebrowser/qutebrowser/browser/hints.py

711 lines
27 KiB
Python
Raw Normal View History

2014-06-19 09:04:37 +02:00
# vim: ft=python fileencoding=utf-8 sts=4 sw=4 et:
2014-04-19 17:50:11 +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/>.
"""A HintManager to draw hints over links."""
2014-04-20 23:58:14 +02:00
import math
2014-07-29 01:45:42 +02:00
import subprocess
2014-08-26 19:10:14 +02:00
import collections
2014-04-20 19:24:22 +02:00
from PyQt5.QtCore import pyqtSignal, pyqtSlot, QObject, QEvent, Qt, QUrl
from PyQt5.QtGui import QMouseEvent, QClipboard
from PyQt5.QtWidgets import QApplication
2014-04-21 15:20:41 +02:00
2014-08-26 19:10:14 +02:00
from qutebrowser.config import config
from qutebrowser.keyinput import modeman
2014-09-08 10:30:05 +02:00
from qutebrowser.browser import webelem
2014-09-25 07:44:11 +02:00
from qutebrowser.commands import userscripts, cmdexc, cmdutils
from qutebrowser.utils import usertypes, log, qtutils, message, objreg
2014-04-21 15:20:41 +02:00
2014-09-28 00:43:08 +02:00
ElemTuple = collections.namedtuple('ElemTuple', ['elem', 'label'])
2014-04-21 15:45:29 +02:00
Target = usertypes.enum('Target', ['normal', 'tab', 'tab_bg', 'yank',
'yank_primary', 'fill', 'rapid', 'download',
'userscript', 'spawn'])
2014-05-05 07:45:36 +02:00
@pyqtSlot(usertypes.KeyMode)
2014-09-28 22:13:14 +02:00
def on_mode_entered(mode, win_id):
"""Stop hinting when insert mode was entered."""
if mode == usertypes.KeyMode.insert:
2014-09-28 22:13:14 +02:00
modeman.maybe_leave(win_id, usertypes.KeyMode.hint, 'insert mode')
class HintContext:
"""Context namespace used for hinting.
Attributes:
frames: The QWebFrames to use.
elems: A mapping from keystrings to (elem, label) namedtuples.
baseurl: The URL of the current page.
target: What to do with the opened links.
normal/tab/tab_bg: Get passed to BrowserTab.
yank/yank_primary: Yank to clipboard/primary selection
fill: Fill commandline with link.
rapid: Rapid mode with background tabs
2014-06-19 17:58:46 +02:00
download: Download the link.
2014-07-29 01:45:42 +02:00
userscript: Call a custom userscript.
spawn: Spawn a simple command.
to_follow: The link to follow when enter is pressed.
connected_frames: The QWebFrames which are connected to a signal.
2014-07-29 01:45:42 +02:00
args: Custom arguments for userscript/spawn
"""
def __init__(self):
self.elems = {}
self.target = None
self.baseurl = None
self.to_follow = None
self.frames = []
self.connected_frames = []
2014-07-29 01:45:42 +02:00
self.args = []
def get_args(self, urlstr):
"""Get the arguments, with {hint-url} replaced by the given URL."""
args = []
for arg in self.args:
if arg == '{hint-url}':
args.append(urlstr)
else:
args.append(arg)
return args
2014-04-21 15:20:41 +02:00
class HintManager(QObject):
2014-04-19 17:50:11 +02:00
"""Manage drawing hints over links or other elements.
Class attributes:
2014-04-20 23:03:18 +02:00
HINT_CSS: The CSS template to use for hints.
2014-05-13 21:13:53 +02:00
HINT_TEXTS: Text displayed for different hinting modes.
2014-04-19 17:50:11 +02:00
Attributes:
_context: The HintContext for the current invocation.
2014-09-28 22:13:14 +02:00
_win_id: The window ID this HintManager is associated with.
_tab_id: The tab ID this HintManager is associated with.
2014-04-21 15:20:41 +02:00
Signals:
2014-04-21 16:59:03 +02:00
mouse_event: Mouse event to be posted in the web view.
arg: A QMouseEvent
2014-04-21 19:29:11 +02:00
set_open_target: Set a new target to open the links in.
2014-04-19 17:50:11 +02:00
"""
2014-04-19 23:16:39 +02:00
HINT_CSS = """
2014-05-06 17:02:32 +02:00
display: {display};
2014-04-20 19:24:22 +02:00
color: {config[colors][hints.fg]};
background: {config[colors][hints.bg]};
font: {config[fonts][hints]};
border: {config[hints][border]};
opacity: {config[hints][opacity]};
2014-04-20 16:42:55 +02:00
z-index: 100000;
2014-04-21 23:32:58 +02:00
pointer-events: none;
2014-04-19 23:16:39 +02:00
position: absolute;
left: {left}px;
top: {top}px;
"""
2014-05-13 21:13:53 +02:00
HINT_TEXTS = {
Target.normal: "Follow hint...",
Target.tab: "Follow hint in new tab...",
2014-05-16 23:01:40 +02:00
Target.tab_bg: "Follow hint in background tab...",
2014-05-13 21:13:53 +02:00
Target.yank: "Yank hint to clipboard...",
Target.yank_primary: "Yank hint to primary selection...",
Target.fill: "Set hint in commandline...",
2014-05-13 21:13:53 +02:00
Target.rapid: "Follow hint (rapid mode)...",
2014-06-19 17:58:46 +02:00
Target.download: "Download hint...",
2014-07-29 01:45:42 +02:00
Target.userscript: "Call userscript via hint...",
Target.spawn: "Spawn command via hint...",
2014-05-13 21:13:53 +02:00
}
2014-04-21 16:59:03 +02:00
mouse_event = pyqtSignal('QMouseEvent')
2014-04-21 19:29:11 +02:00
set_open_target = pyqtSignal(str)
2014-04-21 15:20:41 +02:00
def __init__(self, win_id, tab_id, parent=None):
2014-09-28 22:13:14 +02:00
"""Constructor."""
super().__init__(parent)
2014-09-28 22:13:14 +02:00
self._win_id = win_id
self._tab_id = tab_id
self._context = None
2014-09-28 22:13:14 +02:00
mode_manager = objreg.get('mode-manager', scope='window',
window=win_id)
mode_manager.left.connect(self.on_mode_left)
2014-04-19 23:16:39 +02:00
2014-08-04 03:14:14 +02:00
def _cleanup(self):
"""Clean up after hinting."""
for elem in self._context.elems.values():
2014-09-04 08:00:05 +02:00
try:
2014-08-04 03:14:14 +02:00
elem.label.removeFromDocument()
2014-09-04 08:00:05 +02:00
except webelem.IsNullError:
pass
text = self.HINT_TEXTS[self._context.target]
2014-09-28 22:13:14 +02:00
message_bridge = objreg.get('message-bridge', scope='window',
window=self._win_id)
message_bridge.maybe_reset_text(text)
2014-08-04 03:14:14 +02:00
self._context = None
2014-04-20 23:58:14 +02:00
def _hint_strings(self, elems):
"""Calculate the hint strings for elems.
2014-04-22 08:45:56 +02:00
Inspired by Vimium.
2014-04-21 00:24:08 +02:00
Args:
elems: The elements to get hint strings for.
Return:
A list of hint strings, in the same order as the elements.
2014-04-20 23:58:14 +02:00
"""
2014-05-02 17:53:16 +02:00
if config.get('hints', 'mode') == 'number':
chars = '0123456789'
else:
chars = config.get('hints', 'chars')
2014-04-20 23:58:14 +02:00
# Determine how many digits the link hints will require in the worst
# case. Usually we do not need all of these digits for every link
# single hint, so we can show shorter hints for a few of the links.
needed = math.ceil(math.log(len(elems), len(chars)))
# Short hints are the number of hints we can possibly show which are
# (needed - 1) digits in length.
short_count = math.floor((len(chars) ** needed - len(elems)) /
len(chars))
long_count = len(elems) - short_count
strings = []
if needed > 1:
for i in range(short_count):
strings.append(self._number_to_hint_str(i, chars, needed - 1))
start = short_count * len(chars)
for i in range(start, start + long_count):
strings.append(self._number_to_hint_str(i, chars, needed))
return self._shuffle_hints(strings, len(chars))
def _shuffle_hints(self, hints, length):
"""Shuffle the given set of hints so that they're scattered.
Hints starting with the same character will be spread evenly throughout
the array.
Inspired by Vimium.
2014-04-21 00:24:08 +02:00
Args:
hints: A list of hint strings.
length: Length of the available charset.
Return:
A list of shuffled hint strings.
2014-04-20 23:58:14 +02:00
"""
buckets = [[] for i in range(length)]
for i, hint in enumerate(hints):
buckets[i % len(buckets)].append(hint)
result = []
for bucket in buckets:
result += bucket
return result
def _number_to_hint_str(self, number, chars, digits=0):
"""Convert a number like "8" into a hint string like "JK".
This is used to sequentially generate all of the hint text.
The hint string will be "padded with zeroes" to ensure its length is >=
digits.
Inspired by Vimium.
2014-04-21 00:24:08 +02:00
Args:
number: The hint number.
chars: The charset to use.
digits: The minimum output length.
Return:
A hint string.
2014-04-20 23:58:14 +02:00
"""
base = len(chars)
hintstr = []
remainder = 0
while True:
remainder = number % base
hintstr.insert(0, chars[remainder])
number -= remainder
number //= base
if number <= 0:
break
# Pad the hint string we're returning so that it matches digits.
2014-04-21 00:24:08 +02:00
for _ in range(0, digits - len(hintstr)):
2014-04-20 23:58:14 +02:00
hintstr.insert(0, chars[0])
return ''.join(hintstr)
2014-05-06 17:02:32 +02:00
def _get_hint_css(self, elem, label=None):
"""Get the hint CSS for the element given.
Args:
elem: The QWebElement to get the CSS for.
label: The label QWebElement if display: none should be preserved.
Return:
The CSS to set as a string.
"""
2014-09-04 08:00:05 +02:00
if label is None or label['hidden'] != 'true':
2014-05-06 17:02:32 +02:00
display = 'inline'
else:
display = 'none'
rect = elem.geometry()
2014-09-23 23:17:36 +02:00
return self.HINT_CSS.format(
2014-09-24 07:06:45 +02:00
left=rect.x(), top=rect.y(), config=objreg.get('config'),
2014-09-23 23:17:36 +02:00
display=display)
2014-05-06 17:02:32 +02:00
2014-04-20 23:58:14 +02:00
def _draw_label(self, elem, string):
2014-04-21 00:24:08 +02:00
"""Draw a hint label over an element.
Args:
elem: The QWebElement to use.
string: The hint string to print.
2014-04-21 15:45:29 +02:00
Return:
The newly created label elment
2014-04-21 00:24:08 +02:00
"""
2014-05-06 17:02:32 +02:00
css = self._get_hint_css(elem)
2014-05-12 07:49:44 +02:00
doc = elem.webFrame().documentElement()
2014-04-23 14:34:00 +02:00
# It seems impossible to create an empty QWebElement for which isNull()
# is false so we can work with it.
# As a workaround, we use appendInside() with markup as argument, and
# then use lastChild() to get a reference to it.
# See: http://stackoverflow.com/q/7364852/2085149
2014-04-20 23:58:14 +02:00
doc.appendInside('<span class="qutehint" style="{}">{}</span>'.format(
2014-04-22 09:35:49 +02:00
css, string))
2014-09-04 08:00:05 +02:00
elem = webelem.WebElementWrapper(doc.lastChild())
elem['hidden'] = 'false'
return elem
2014-04-19 17:50:11 +02:00
2014-05-01 15:29:18 +02:00
def _click(self, elem):
"""Click an element.
Args:
elem: The QWebElement to click.
"""
if self._context.target == Target.rapid:
2014-05-16 23:01:40 +02:00
target = Target.tab_bg
else:
target = self._context.target
self.set_open_target.emit(target.name)
# FIXME Instead of clicking the center, we could have nicer heuristics.
# e.g. parse (-webkit-)border-radius correctly and click text fields at
# the bottom right, and everything else on the top left or so.
2014-09-04 08:00:05 +02:00
pos = elem.rect_on_view().center()
log.hints.debug("Clicking on '{}' at {}/{}".format(
elem, pos.x(), pos.y()))
events = (
2014-05-12 07:49:44 +02:00
QMouseEvent(QEvent.MouseMove, pos, Qt.NoButton, Qt.NoButton,
Qt.NoModifier),
2014-05-12 07:49:44 +02:00
QMouseEvent(QEvent.MouseButtonPress, pos, Qt.LeftButton,
Qt.NoButton, Qt.NoModifier),
2014-05-12 07:49:44 +02:00
QMouseEvent(QEvent.MouseButtonRelease, pos, Qt.LeftButton,
Qt.NoButton, Qt.NoModifier),
)
for evt in events:
self.mouse_event.emit(evt)
def _yank(self, url):
"""Yank an element to the clipboard or primary selection.
Args:
url: The URL to open as a QURL.
"""
qtutils.ensure_valid(url)
sel = self._context.target == Target.yank_primary
mode = QClipboard.Selection if sel else QClipboard.Clipboard
urlstr = url.toString(QUrl.FullyEncoded | QUrl.RemovePassword)
QApplication.clipboard().setText(urlstr, mode)
2014-09-28 22:13:14 +02:00
message.info(self._win_id, "URL yanked to {}".format(
"primary selection" if sel else "clipboard"))
def _preset_cmd_text(self, url):
"""Preset a commandline text based on a hint URL.
Args:
url: The URL to open as a QUrl.
"""
qtutils.ensure_valid(url)
urlstr = url.toDisplayString(QUrl.FullyEncoded)
args = self._context.get_args(urlstr)
2014-09-28 22:13:14 +02:00
message.set_cmd_text(self._win_id, ' '.join(args))
def _download(self, elem):
2014-06-19 17:58:46 +02:00
"""Download a hint URL.
Args:
elem: The QWebElement to download.
2014-06-19 17:58:46 +02:00
"""
url = self._resolve_url(elem)
if url is None:
2014-09-28 22:13:14 +02:00
message.error(self._win_id,
"No suitable link found for this element.",
immediately=True)
return
qtutils.ensure_valid(url)
2014-09-25 07:58:08 +02:00
objreg.get('download-manager').get(url, elem.webFrame().page())
2014-06-19 17:58:46 +02:00
2014-07-29 01:45:42 +02:00
def _call_userscript(self, url):
"""Call an userscript from a hint."""
qtutils.ensure_valid(url)
2014-07-29 02:05:15 +02:00
cmd = self._context.args[0]
args = self._context.args[1:]
2014-09-28 22:23:37 +02:00
userscripts.run(cmd, *args, url=url, win_id=self._win_id)
2014-07-29 01:45:42 +02:00
def _spawn(self, url):
"""Spawn a simple command from a hint."""
qtutils.ensure_valid(url)
2014-07-29 01:45:42 +02:00
urlstr = url.toString(QUrl.FullyEncoded | QUrl.RemovePassword)
args = self._context.get_args(urlstr)
2014-07-29 01:45:42 +02:00
subprocess.Popen(args)
def _resolve_url(self, elem, baseurl=None):
"""Resolve a URL and check if we want to keep it.
2014-04-21 23:53:13 +02:00
Args:
elem: The QWebElement to get the URL of.
baseurl: The baseurl of the current tab (overrides baseurl from
self._context).
2014-04-21 23:53:13 +02:00
Return:
A QUrl with the absolute URL, or None.
2014-04-21 23:53:13 +02:00
"""
try:
text = elem['href']
except KeyError:
2014-04-21 23:53:13 +02:00
return None
url = QUrl(text)
if not url.isValid():
return None
if url.isRelative():
2014-09-15 06:53:05 +02:00
if baseurl is None:
baseurl = self._context.baseurl
url = baseurl.resolved(url)
return url
2014-04-21 23:53:13 +02:00
2014-05-01 16:35:26 +02:00
def _find_prevnext(self, frame, prev=False):
"""Find a prev/next element in frame."""
2014-04-30 18:01:03 +02:00
# First check for <link rel="prev(ious)|next">
2014-05-06 21:09:09 +02:00
elems = frame.findAllElements(
2014-07-16 09:17:59 +02:00
webelem.SELECTORS[webelem.Group.links])
rel_values = ('prev', 'previous') if prev else ('next')
2014-04-30 18:01:03 +02:00
for e in elems:
2014-09-04 08:00:05 +02:00
e = webelem.WebElementWrapper(e)
try:
rel_attr = e['rel']
except KeyError:
continue
if rel_attr in rel_values:
log.hints.debug("Found '{}' with rel={}".format(
2014-09-04 08:00:05 +02:00
e.debug_text(), rel_attr))
2014-05-01 16:35:26 +02:00
return e
2014-07-16 09:17:59 +02:00
# Then check for regular links/buttons.
2014-05-06 21:09:09 +02:00
elems = frame.findAllElements(
webelem.SELECTORS[webelem.Group.prevnext])
2014-04-30 18:01:03 +02:00
option = 'prev-regexes' if prev else 'next-regexes'
2014-05-01 16:35:26 +02:00
if not elems:
return None
for regex in config.get('hints', option):
log.hints.vdebug("== Checking regex '{}'.".format(regex.pattern))
2014-05-01 16:35:26 +02:00
for e in elems:
2014-09-04 08:00:05 +02:00
e = webelem.WebElementWrapper(e)
text = str(e)
if not text:
continue
if regex.search(text):
log.hints.debug("Regex '{}' matched on '{}'.".format(
regex.pattern, text))
2014-05-01 16:35:26 +02:00
return e
else:
log.hints.vdebug("No match on '{}'!".format(text))
2014-05-01 16:35:26 +02:00
return None
2014-05-13 21:13:53 +02:00
def _connect_frame_signals(self):
"""Connect the contentsSizeChanged signals to all frames."""
for f in self._context.frames:
2014-05-13 21:13:53 +02:00
# For some reason we get segfaults sometimes when calling
# frame.contentsSizeChanged.disconnect() later, maybe because Qt
# already deleted the frame?
# We work around this by never disconnecting this signal, and here
# making sure we don't connect a frame which already was connected
# at some point earlier.
if f in self._context.connected_frames:
2014-08-26 20:15:41 +02:00
log.hints.debug("Frame {} already connected!".format(f))
2014-05-13 21:13:53 +02:00
else:
2014-08-26 20:15:41 +02:00
log.hints.debug("Connecting frame {}".format(f))
2014-05-13 21:13:53 +02:00
f.contentsSizeChanged.connect(self.on_contents_size_changed)
self._context.connected_frames.append(f)
2014-05-13 21:13:53 +02:00
2014-09-14 23:09:01 +02:00
def _check_args(self, target, *args):
2014-09-05 06:55:53 +02:00
"""Check the arguments passed to start() and raise if they're wrong.
Args:
target: A Target enum member.
args: Arguments for userscript/download
"""
if not isinstance(target, Target):
raise TypeError("Target {} is no Target member!".format(target))
if target in (Target.userscript, Target.spawn, Target.fill):
2014-09-14 23:09:01 +02:00
if not args:
2014-09-05 06:55:53 +02:00
raise cmdexc.CommandError(
"'args' is required with target userscript/spawn/fill.")
else:
2014-09-14 23:09:01 +02:00
if args:
2014-09-05 06:55:53 +02:00
raise cmdexc.CommandError(
"'args' is only allowed with target userscript/spawn.")
def _init_elements(self, mainframe, group):
"""Initialize the elements and labels based on the context set.
Args:
mainframe: The main QWebFrame.
group: A Group enum member (which elements to find).
"""
elems = []
for f in self._context.frames:
elems += f.findAllElements(webelem.SELECTORS[group])
elems = [e for e in elems if webelem.is_visible(e, mainframe)]
# We wrap the elements late for performance reasons, as wrapping 1000s
# of elements (with ~50 methods each) just takes too much time...
elems = [webelem.WebElementWrapper(e) for e in elems]
2014-09-05 06:55:53 +02:00
filterfunc = webelem.FILTERS.get(group, lambda e: True)
elems = [e for e in elems if filterfunc(e)]
if not elems:
2014-09-05 06:55:53 +02:00
raise cmdexc.CommandError("No elements found.")
strings = self._hint_strings(elems)
for e, string in zip(elems, strings):
2014-09-05 06:55:53 +02:00
label = self._draw_label(e, string)
self._context.elems[string] = ElemTuple(e, label)
2014-09-28 23:23:02 +02:00
keyparsers = objreg.get('keyparsers', scope='window',
window=self._win_id)
keyparser = keyparsers[usertypes.KeyMode.hint]
2014-09-25 07:44:11 +02:00
keyparser.update_bindings(strings)
2014-09-05 06:55:53 +02:00
2014-05-01 16:40:14 +02:00
def follow_prevnext(self, frame, baseurl, prev=False, newtab=False):
2014-05-01 16:35:26 +02:00
"""Click a "previous"/"next" element on the page.
Args:
frame: The frame where the element is in.
2014-05-01 16:40:14 +02:00
baseurl: The base URL of the current tab.
2014-05-01 16:35:26 +02:00
prev: True to open a "previous" link, False to open a "next" link.
newtab: True to open in a new tab, False for the current tab.
"""
elem = self._find_prevnext(frame, prev)
if elem is None:
2014-08-26 19:10:14 +02:00
raise cmdexc.CommandError("No {} links found!".format(
2014-05-14 18:00:40 +02:00
"prev" if prev else "forward"))
url = self._resolve_url(elem, baseurl)
if url is None:
2014-08-26 19:10:14 +02:00
raise cmdexc.CommandError("No {} links found!".format(
2014-05-14 18:00:40 +02:00
"prev" if prev else "forward"))
2014-09-25 07:44:11 +02:00
qtutils.ensure_valid(url)
if newtab:
2014-09-28 23:23:02 +02:00
tabbed_browser = objreg.get('tabbed-browser', scope='window',
window=self._win_id)
tabbed_browser.tabopen(url, background=False)
2014-09-25 07:44:11 +02:00
else:
webview = objreg.get('webview', scope='tab', window=self._win_id,
tab=self._tab_id)
webview.openurl(url)
2014-04-30 18:01:03 +02:00
2014-09-25 07:44:11 +02:00
@cmdutils.register(instance='hintmanager', scope='tab', name='hint')
def start(self, group=webelem.Group.all, target=Target.normal,
*args: {'nargs': '*'}):
2014-04-19 17:50:11 +02:00
"""Start hinting.
Args:
2014-09-25 07:44:11 +02:00
group: The hinting mode to use.
- `all`: All clickable elements.
- `links`: Only links.
- `images`: Only images.
target: What to do with the selected element.
- `normal`: Open the link in the current tab.
- `tab`: Open the link in a new tab.
- `tab-bg`: Open the link in a new background tab.
- `yank`: Yank the link to the clipboard.
- `yank-primary`: Yank the link to the primary selection.
- `fill`: Fill the commandline with the command given as
argument.
- `rapid`: Open the link in a new tab and stay in hinting mode.
- `download`: Download the link.
- `userscript`: Call an userscript with `$QUTE_URL` set to the
link.
- `spawn`: Spawn a command.
*args: Arguments for spawn/userscript/fill.
- With `spawn`: The executable and arguments to spawn.
`{hint-url}` will get replaced by the selected
URL.
- With `userscript`: The userscript to execute.
- With `fill`: The command to fill the statusbar with.
`{hint-url}` will get replaced by the selected
URL.
2014-04-19 17:50:11 +02:00
"""
2014-09-28 23:23:02 +02:00
tabbed_browser = objreg.get('tabbed-browser', scope='window',
window=self._win_id)
2014-09-25 07:44:11 +02:00
widget = tabbed_browser.currentWidget()
if widget is None:
raise cmdexc.CommandError("No WebView available yet!")
mainframe = widget.page().mainFrame()
if mainframe is None:
2014-09-25 07:44:11 +02:00
raise cmdexc.CommandError("No frame focused!")
self._check_args(target, *args)
2014-09-05 06:55:53 +02:00
self._context = HintContext()
self._context.target = target
2014-09-25 07:44:11 +02:00
self._context.baseurl = tabbed_browser.current_url()
2014-09-05 06:55:53 +02:00
self._context.frames = webelem.get_child_frames(mainframe)
2014-09-14 23:09:01 +02:00
self._context.args = args
2014-09-05 06:55:53 +02:00
self._init_elements(mainframe, group)
2014-09-28 23:23:02 +02:00
message_bridge = objreg.get('message-bridge', scope='window',
window=self._win_id)
message_bridge.set_text(self.HINT_TEXTS[target])
2014-05-13 21:13:53 +02:00
self._connect_frame_signals()
2014-08-04 03:14:14 +02:00
try:
2014-09-28 22:13:14 +02:00
modeman.enter(self._win_id, usertypes.KeyMode.hint,
'HintManager.start')
2014-08-04 03:14:14 +02:00
except modeman.ModeLockedError:
self._cleanup()
2014-04-20 17:25:46 +02:00
2014-04-21 15:20:41 +02:00
def handle_partial_key(self, keystr):
"""Handle a new partial keypress."""
2014-08-26 20:15:41 +02:00
log.hints.debug("Handling new keystring: '{}'".format(keystr))
for (string, elems) in self._context.elems.items():
2014-04-21 15:45:29 +02:00
if string.startswith(keystr):
matched = string[:len(keystr)]
rest = string[len(keystr):]
elems.label.setInnerXml('<font color="{}">{}</font>{}'.format(
2014-04-25 16:53:23 +02:00
config.get('colors', 'hints.fg.match'), matched, rest))
2014-09-04 08:00:05 +02:00
if elems.label['hidden'] == 'true':
2014-05-06 17:02:32 +02:00
# hidden element which matches again -> unhide it
2014-09-04 08:00:05 +02:00
elems.label['hidden'] = 'false'
2014-05-06 17:02:32 +02:00
css = self._get_hint_css(elems.elem, elems.label)
2014-09-04 08:00:05 +02:00
elems.label['style'] = css
2014-04-21 15:45:29 +02:00
else:
2014-05-06 17:02:32 +02:00
# element doesn't match anymore -> hide it
2014-09-04 08:00:05 +02:00
elems.label['hidden'] = 'true'
2014-05-06 17:02:32 +02:00
css = self._get_hint_css(elems.elem, elems.label)
2014-09-04 08:00:05 +02:00
elems.label['style'] = css
2014-04-21 15:20:41 +02:00
2014-05-02 17:53:16 +02:00
def filter_hints(self, filterstr):
"""Filter displayed hints according to a text."""
for elems in self._context.elems.values():
2014-09-04 08:00:05 +02:00
if str(elems.elem).lower().startswith(filterstr):
if elems.label['hidden'] == 'true':
2014-05-06 17:02:32 +02:00
# hidden element which matches again -> unhide it
2014-09-04 08:00:05 +02:00
elems.label['hidden'] = 'false'
2014-05-06 17:02:32 +02:00
css = self._get_hint_css(elems.elem, elems.label)
2014-09-04 08:00:05 +02:00
elems.label['style'] = css
2014-05-06 17:02:32 +02:00
else:
# element doesn't match anymore -> hide it
2014-09-04 08:00:05 +02:00
elems.label['hidden'] = 'true'
2014-05-06 17:02:32 +02:00
css = self._get_hint_css(elems.elem, elems.label)
2014-09-04 08:00:05 +02:00
elems.label['style'] = css
2014-05-06 17:02:32 +02:00
visible = {}
for k, e in self._context.elems.items():
2014-09-04 08:00:05 +02:00
if e.label['hidden'] != 'true':
2014-05-06 17:02:32 +02:00
visible[k] = e
if not visible:
2014-05-02 17:53:16 +02:00
# Whoops, filtered all hints
2014-09-28 22:13:14 +02:00
modeman.leave(self._win_id, usertypes.KeyMode.hint, 'all filtered')
2014-05-06 17:02:32 +02:00
elif len(visible) == 1 and config.get('hints', 'auto-follow'):
2014-05-02 17:53:16 +02:00
# unpacking gets us the first (and only) key in the dict.
2014-05-06 17:02:32 +02:00
self.fire(*visible)
2014-05-02 17:53:16 +02:00
2014-04-27 21:59:23 +02:00
def fire(self, keystr, force=False):
"""Fire a completed hint.
Args:
keystr: The keychain string to follow.
force: When True, follow even when auto-follow is false.
"""
if not (force or config.get('hints', 'auto-follow')):
self.handle_partial_key(keystr)
self._context.to_follow = keystr
2014-04-27 21:59:23 +02:00
return
# Handlers which take a QWebElement
elem_handlers = {
2014-05-05 07:45:36 +02:00
Target.normal: self._click,
Target.tab: self._click,
2014-05-16 23:01:40 +02:00
Target.tab_bg: self._click,
2014-05-05 07:45:36 +02:00
Target.rapid: self._click,
# _download needs a QWebElement to get the frame.
Target.download: self._download,
}
# Handlers which take a QUrl
url_handlers = {
2014-05-05 07:45:36 +02:00
Target.yank: self._yank,
Target.yank_primary: self._yank,
Target.fill: self._preset_cmd_text,
2014-07-29 01:45:42 +02:00
Target.userscript: self._call_userscript,
Target.spawn: self._spawn,
}
elem = self._context.elems[keystr].elem
if self._context.target in elem_handlers:
elem_handlers[self._context.target](elem)
elif self._context.target in url_handlers:
url = self._resolve_url(elem)
if url is None:
2014-09-28 22:13:14 +02:00
message.error(self._win_id,
"No suitable link found for this element.",
2014-06-26 07:58:00 +02:00
immediately=True)
return
url_handlers[self._context.target](url)
2014-05-05 07:45:36 +02:00
else:
raise ValueError("No suitable handler found!")
if self._context.target != Target.rapid:
2014-09-28 22:13:14 +02:00
modeman.maybe_leave(self._win_id, usertypes.KeyMode.hint,
'followed')
2014-09-25 07:44:11 +02:00
@cmdutils.register(instance='hintmanager', scope='tab', hide=True)
2014-04-27 21:59:23 +02:00
def follow_hint(self):
2014-04-28 00:05:14 +02:00
"""Follow the currently selected hint."""
if not self._context.to_follow:
2014-08-26 19:10:14 +02:00
raise cmdexc.CommandError("No hint to follow")
self.fire(self._context.to_follow, force=True)
2014-04-27 21:59:23 +02:00
@pyqtSlot('QSize')
2014-04-22 17:53:27 +02:00
def on_contents_size_changed(self, _size):
"""Reposition hints if contents size changed."""
if self._context is None:
2014-05-13 13:42:30 +02:00
# We got here because of some earlier hinting, but we can't simply
# disconnect frames as this leads to occasional segfaults :-/
2014-08-26 20:15:41 +02:00
log.hints.debug("Not hinting!")
2014-05-13 13:42:30 +02:00
return
2014-08-26 20:15:41 +02:00
log.hints.debug("Contents size changed...!")
for elems in self._context.elems.values():
2014-05-06 17:02:32 +02:00
css = self._get_hint_css(elems.elem, elems.label)
2014-09-04 08:00:05 +02:00
elems.label['style'] = css
2014-04-24 23:47:02 +02:00
2014-08-26 19:10:14 +02:00
@pyqtSlot(usertypes.KeyMode)
2014-04-24 23:47:02 +02:00
def on_mode_left(self, mode):
"""Stop hinting when hinting mode was left."""
2014-08-26 19:10:14 +02:00
if mode != usertypes.KeyMode.hint or self._context is None:
# We have one HintManager per tab, so when this gets called,
# self._context might be None, because the current tab is not
# hinting.
2014-04-24 23:47:02 +02:00
return
2014-08-04 03:14:14 +02:00
self._cleanup()