hints: split getClientRects() into separate method

this will be useful for positioning the hint label
This commit is contained in:
Jakub Klinkovský 2016-02-15 22:24:08 +01:00
parent 239b7497e9
commit dd594b0eca

View File

@ -26,7 +26,7 @@ import re
import string import string
from PyQt5.QtCore import (pyqtSignal, pyqtSlot, QObject, QEvent, Qt, QUrl, from PyQt5.QtCore import (pyqtSignal, pyqtSlot, QObject, QEvent, Qt, QUrl,
QTimer, QPoint) QTimer, QRect)
from PyQt5.QtGui import QMouseEvent from PyQt5.QtGui import QMouseEvent
from PyQt5.QtWebKit import QWebElement from PyQt5.QtWebKit import QWebElement
from PyQt5.QtWebKitWidgets import QWebPage from PyQt5.QtWebKitWidgets import QWebPage
@ -420,6 +420,32 @@ class HintManager(QObject):
message.error(self._win_id, "No suitable link found for this element.", message.error(self._win_id, "No suitable link found for this element.",
immediately=True) immediately=True)
def _get_first_rectangle(self, elem):
"""Return the element's first client rectangle with positive size.
Uses the getClientRects() JavaScript method to obtain the collection of
rectangles containing the element and returns the first with positive
dimensions. Falls back to elem.rect_on_view() in case all rectangles
returned by getClientRects() have zero dimensions.
Skipping of rectangles with zero dimensions is due to <a> elements
containing other elements with "display:block" style, see
https://github.com/The-Compiler/qutebrowser/issues/1298
Args:
elem: The QWebElement of interest.
"""
rects = elem.evaluateJavaScript("this.getClientRects()")
log.hints.debug("Client rectangles of element '{}': {}"
.format(elem.debug_text(), rects))
for i in range(int(rects.get("length", 0))):
rect = rects[str(i)]
width = rect.get("width", 0)
height = rect.get("height", 0)
if width > 0 and height > 0:
return QRect(rect["left"], rect["top"], width, height)
return elem.rect_on_view()
def _click(self, elem, context): def _click(self, elem, context):
"""Click an element. """Click an element.
@ -439,29 +465,16 @@ class HintManager(QObject):
else: else:
target_mapping[Target.tab] = usertypes.ClickTarget.tab target_mapping[Target.tab] = usertypes.ClickTarget.tab
action = "Hovering" if context.target == Target.hover else "Clicking"
log.hints.debug("{} on element '{}'".format(action, elem.debug_text()))
# FIXME Instead of clicking the center, we could have nicer heuristics. # FIXME Instead of clicking the center, we could have nicer heuristics.
# e.g. parse (-webkit-)border-radius correctly and click text fields at # e.g. parse (-webkit-)border-radius correctly and click text fields at
# the bottom right, and everything else on the top left or so. # the bottom right, and everything else on the top left or so.
# https://github.com/The-Compiler/qutebrowser/issues/70 # https://github.com/The-Compiler/qutebrowser/issues/70
pos = elem.rect_on_view().center() rect = self._get_first_rectangle(elem)
log.hints.debug("Center position: {}".format(pos)) pos = rect.center()
boxes = elem.evaluateJavaScript("this.getClientRects()")
log.hints.debug("Bounding boxes: {}".format(boxes)) action = "Hovering" if context.target == Target.hover else "Clicking"
for key in sorted(boxes): log.hints.debug("{} on '{}' at position {}".format(
box = boxes[key] action, elem.debug_text(), pos))
width = box.get("width", 0)
height = box.get("height", 0)
# skip boxes with zero dimensions (happens to <a> if they contain
# other elements with display:block style)
# https://github.com/The-Compiler/qutebrowser/issues/1298
if width > 0 and height > 0:
pos = QPoint(box["left"] + width / 2, box["top"] + height / 2)
log.hints.debug("Updated position: {}".format(pos))
break
log.hints.debug("Final position is {}".format(pos))
self.start_hinting.emit(target_mapping[context.target]) self.start_hinting.emit(target_mapping[context.target])
if context.target in [Target.tab, Target.tab_fg, Target.tab_bg, if context.target in [Target.tab, Target.tab_fg, Target.tab_bg,