# vim: ft=python fileencoding=utf-8 sts=4 sw=4 et: # Copyright 2014-2015 Florian Bruhin (The Compiler) # # 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 . # pylint: disable=protected-access """Tests for the webelement utils.""" import unittest from unittest import mock import collections.abc from PyQt5.QtCore import QRect, QPoint from PyQt5.QtWebKit import QWebElement from qutebrowser.browser import webelem from qutebrowser.test import stubs def get_webelem(geometry=None, frame=None, null=False, visibility='', display='', attributes=None, tagname=None, classes=None): """Factory for WebElementWrapper objects based on a mock. Args: geometry: The geometry of the QWebElement as QRect. frame: The QWebFrame the element is in. null: Whether the element is null or not. visibility: The CSS visibility style property calue. display: The CSS display style property calue. attributes: Boolean HTML attributes to be added. tagname: The tag name. classes: HTML classes to be added. """ elem = mock.Mock() elem.isNull.return_value = null elem.geometry.return_value = geometry elem.webFrame.return_value = frame elem.tagName.return_value = tagname elem.toOuterXml.return_value = '' if attributes is not None: if not isinstance(attributes, collections.abc.Mapping): attributes = {e: None for e in attributes} elem.hasAttribute.side_effect = lambda k: k in attributes elem.attribute.side_effect = lambda k: attributes.get(k, '') elem.attributeNames.return_value = list(attributes) else: elem.hasAttribute.return_value = False elem.attribute.return_value = '' elem.attributeNames.return_value = [] if classes is not None: elem.classes.return_value = classes.split(' ') else: elem.classes.return_value = [] def _style_property(name, strategy): """Helper function to act as styleProperty method.""" if strategy != QWebElement.ComputedStyle: raise ValueError("styleProperty called with strategy != " "ComputedStyle ({})!".format(strategy)) if name == 'visibility': return visibility elif name == 'display': return display else: raise ValueError("styleProperty called with unknown name " "'{}'".format(name)) elem.styleProperty.side_effect = _style_property wrapped = webelem.WebElementWrapper(elem) if attributes is not None: wrapped.update(attributes) return wrapped class WebElementWrapperTests(unittest.TestCase): """Test WebElementWrapper.""" def test_nullelem(self): """Test __init__ with a null element.""" with self.assertRaises(webelem.IsNullError): get_webelem(null=True) class IsVisibleInvalidTests(unittest.TestCase): """Tests for is_visible with invalid elements. Attributes: frame: The FakeWebFrame we're using to test. """ def setUp(self): self.frame = stubs.FakeWebFrame(QRect(0, 0, 100, 100)) def test_nullelem(self): """Passing an element with isNull() == True. geometry() and webFrame() should not be called, and ValueError should be raised. """ elem = get_webelem() elem._elem.isNull.return_value = True with self.assertRaises(webelem.IsNullError): elem.is_visible(self.frame) def test_invalid_invisible(self): """Test elements with an invalid geometry which are invisible.""" elem = get_webelem(QRect(0, 0, 0, 0), self.frame) self.assertFalse(elem.geometry().isValid()) self.assertEqual(elem.geometry().x(), 0) self.assertFalse(elem.is_visible(self.frame)) def test_invalid_visible(self): """Test elements with an invalid geometry which are visible. This seems to happen sometimes in the real world, with real elements which *are* visible, but don't have a valid geometry. """ elem = get_webelem(QRect(10, 10, 0, 0), self.frame) self.assertFalse(elem.geometry().isValid()) self.assertTrue(elem.is_visible(self.frame)) class IsVisibleScrollTests(unittest.TestCase): """Tests for is_visible when the frame is scrolled. Attributes: frame: The FakeWebFrame we're using to test. """ def setUp(self): self.frame = stubs.FakeWebFrame(QRect(0, 0, 100, 100), scroll=QPoint(10, 10)) def test_invisible(self): """Test elements which should be invisible due to scrolling.""" elem = get_webelem(QRect(5, 5, 4, 4), self.frame) self.assertFalse(elem.is_visible(self.frame)) def test_visible(self): """Test elements which still should be visible after scrolling.""" elem = get_webelem(QRect(10, 10, 1, 1), self.frame) self.assertTrue(elem.is_visible(self.frame)) class IsVisibleCssTests(unittest.TestCase): """Tests for is_visible with CSS attributes. Attributes: frame: The FakeWebFrame we're using to test. """ def setUp(self): self.frame = stubs.FakeWebFrame(QRect(0, 0, 100, 100)) def test_visibility_visible(self): """Check that elements with "visibility = visible" are visible.""" elem = get_webelem(QRect(0, 0, 10, 10), self.frame, visibility='visible') self.assertTrue(elem.is_visible(self.frame)) def test_visibility_hidden(self): """Check that elements with "visibility = hidden" are not visible.""" elem = get_webelem(QRect(0, 0, 10, 10), self.frame, visibility='hidden') self.assertFalse(elem.is_visible(self.frame)) def test_display_inline(self): """Check that elements with "display = inline" are visible.""" elem = get_webelem(QRect(0, 0, 10, 10), self.frame, display='inline') self.assertTrue(elem.is_visible(self.frame)) def test_display_none(self): """Check that elements with "display = none" are not visible.""" elem = get_webelem(QRect(0, 0, 10, 10), self.frame, display='none') self.assertFalse(elem.is_visible(self.frame)) class IsVisibleIframeTests(unittest.TestCase): """Tests for is_visible with a child frame. Attributes: frame: The FakeWebFrame we're using to test. iframe: The iframe inside frame. elem1-elem4: FakeWebElements to test. """ def setUp(self): """Set up the following base situation. 0, 0 300, 0 ############################## # # 0,10 # iframe 100,10 # #********** # #*e * elem1: 0, 0 in iframe (visible) #* * # #* e * elem2: 20,90 in iframe (visible) #********** # 0,110 #. .100,110 # #. . # #. e . elem3: 20,150 in iframe (not visible) #.......... # # e elem4: 30, 180 in main frame (visible) # # # frame # ############################## 300, 0 300, 300 """ self.frame = stubs.FakeWebFrame(QRect(0, 0, 300, 300)) self.iframe = stubs.FakeWebFrame(QRect(0, 10, 100, 100), parent=self.frame) self.elem1 = get_webelem(QRect(0, 0, 10, 10), self.iframe) self.elem2 = get_webelem(QRect(20, 90, 10, 10), self.iframe) self.elem3 = get_webelem(QRect(20, 150, 10, 10), self.iframe) self.elem4 = get_webelem(QRect(30, 180, 10, 10), self.frame) def test_not_scrolled(self): """Test base situation.""" self.assertTrue(self.frame.geometry().contains(self.iframe.geometry())) self.assertTrue(self.elem1.is_visible(self.frame)) self.assertTrue(self.elem2.is_visible(self.frame)) self.assertFalse(self.elem3.is_visible(self.frame)) self.assertTrue(self.elem4.is_visible(self.frame)) def test_iframe_scrolled(self): """Scroll iframe down so elem3 gets visible and elem1/elem2 not.""" self.iframe.scrollPosition.return_value = QPoint(0, 100) self.assertFalse(self.elem1.is_visible(self.frame)) self.assertFalse(self.elem2.is_visible(self.frame)) self.assertTrue(self.elem3.is_visible(self.frame)) self.assertTrue(self.elem4.is_visible(self.frame)) def test_mainframe_scrolled_iframe_visible(self): """Scroll mainframe down so iframe is partly visible but elem1 not.""" self.frame.scrollPosition.return_value = QPoint(0, 50) geom = self.frame.geometry().translated(self.frame.scrollPosition()) self.assertFalse(geom.contains(self.iframe.geometry())) self.assertTrue(geom.intersects(self.iframe.geometry())) self.assertFalse(self.elem1.is_visible(self.frame)) self.assertTrue(self.elem2.is_visible(self.frame)) self.assertFalse(self.elem3.is_visible(self.frame)) self.assertTrue(self.elem4.is_visible(self.frame)) def test_mainframe_scrolled_iframe_invisible(self): """Scroll mainframe down so iframe is invisible.""" self.frame.scrollPosition.return_value = QPoint(0, 110) geom = self.frame.geometry().translated(self.frame.scrollPosition()) self.assertFalse(geom.contains(self.iframe.geometry())) self.assertFalse(geom.intersects(self.iframe.geometry())) self.assertFalse(self.elem1.is_visible(self.frame)) self.assertFalse(self.elem2.is_visible(self.frame)) self.assertFalse(self.elem3.is_visible(self.frame)) self.assertTrue(self.elem4.is_visible(self.frame)) class IsWritableTests(unittest.TestCase): """Check is_writable.""" def test_writable(self): """Test a normal element.""" elem = get_webelem() self.assertTrue(elem.is_writable()) def test_disabled(self): """Test a disabled element.""" elem = get_webelem(attributes=['disabled']) self.assertFalse(elem.is_writable()) def test_readonly(self): """Test a readonly element.""" elem = get_webelem(attributes=['readonly']) self.assertFalse(elem.is_writable()) class JavascriptEscapeTests(unittest.TestCase): """Check javascript_escape. Class attributes: STRINGS: A list of (input, output) tuples. """ STRINGS = ( ('foo\\bar', r'foo\\bar'), ('foo\nbar', r'foo\nbar'), ("foo'bar", r"foo\'bar"), ('foo"bar', r'foo\"bar'), ) def test_fake_escape(self): """Test javascript escaping.""" for before, after in self.STRINGS: with self.subTest(before=before): self.assertEqual(webelem.javascript_escape(before), after) class GetChildFramesTests(unittest.TestCase): """Check get_child_frames.""" def test_single_frame(self): """Test get_child_frames with a single frame without children.""" frame = stubs.FakeChildrenFrame() children = webelem.get_child_frames(frame) self.assertEqual(len(children), 1) self.assertIs(children[0], frame) frame.childFrames.assert_called_once_with() def test_one_level(self): r"""Test get_child_frames with one level of children. o parent / \ child1 o o child2 """ child1 = stubs.FakeChildrenFrame() child2 = stubs.FakeChildrenFrame() parent = stubs.FakeChildrenFrame([child1, child2]) children = webelem.get_child_frames(parent) self.assertEqual(len(children), 3) self.assertIs(children[0], parent) self.assertIs(children[1], child1) self.assertIs(children[2], child2) parent.childFrames.assert_called_once_with() child1.childFrames.assert_called_once_with() child2.childFrames.assert_called_once_with() def test_multiple_levels(self): r"""Test get_child_frames with multiple levels of children. o root / \ o o first /\ /\ o o o o second """ second = [stubs.FakeChildrenFrame() for _ in range(4)] first = [stubs.FakeChildrenFrame(second[0:2]), stubs.FakeChildrenFrame(second[2:4])] root = stubs.FakeChildrenFrame(first) children = webelem.get_child_frames(root) self.assertEqual(len(children), 7) self.assertIs(children[0], root) for frame in [root] + first + second: with self.subTest(frame=frame): frame.childFrames.assert_called_once_with() class IsEditableTests(unittest.TestCase): """Tests for is_editable.""" def setUp(self): webelem.config = None def test_input_plain(self): """Test with plain input element.""" elem = get_webelem(tagname='input') self.assertTrue(elem.is_editable()) def test_input_text(self): """Test with text input element.""" elem = get_webelem(tagname='input', attributes={'type': 'text'}) self.assertTrue(elem.is_editable()) def test_input_text_caps(self): """Test with text input element with caps attributes.""" elem = get_webelem(tagname='INPUT', attributes={'TYPE': 'TEXT'}) self.assertTrue(elem.is_editable()) def test_input_email(self): """Test with email input element.""" elem = get_webelem(tagname='input', attributes={'type': 'email'}) self.assertTrue(elem.is_editable()) def test_input_url(self): """Test with url input element.""" elem = get_webelem(tagname='input', attributes={'type': 'url'}) self.assertTrue(elem.is_editable()) def test_input_tel(self): """Test with tel input element.""" elem = get_webelem(tagname='input', attributes={'type': 'tel'}) self.assertTrue(elem.is_editable()) def test_input_number(self): """Test with number input element.""" elem = get_webelem(tagname='input', attributes={'type': 'number'}) self.assertTrue(elem.is_editable()) def test_input_password(self): """Test with password input element.""" elem = get_webelem(tagname='input', attributes={'type': 'password'}) self.assertTrue(elem.is_editable()) def test_input_search(self): """Test with search input element.""" elem = get_webelem(tagname='input', attributes={'type': 'search'}) self.assertTrue(elem.is_editable()) def test_input_button(self): """Button should not be editable.""" elem = get_webelem(tagname='input', attributes={'type': 'button'}) self.assertFalse(elem.is_editable()) def test_input_checkbox(self): """Checkbox should not be editable.""" elem = get_webelem(tagname='input', attributes={'type': 'checkbox'}) self.assertFalse(elem.is_editable()) def test_textarea(self): """Test textarea element.""" elem = get_webelem(tagname='textarea') self.assertTrue(elem.is_editable()) def test_select(self): """Test selectbox.""" elem = get_webelem(tagname='select') self.assertFalse(elem.is_editable()) def test_input_disabled(self): """Test disabled input element.""" elem = get_webelem(tagname='input', attributes={'disabled': None}) self.assertFalse(elem.is_editable()) def test_input_readonly(self): """Test readonly input element.""" elem = get_webelem(tagname='input', attributes={'readonly': None}) self.assertFalse(elem.is_editable()) def test_textarea_disabled(self): """Test disabled textarea element.""" elem = get_webelem(tagname='textarea', attributes={'disabled': None}) self.assertFalse(elem.is_editable()) def test_textarea_readonly(self): """Test readonly textarea element.""" elem = get_webelem(tagname='textarea', attributes={'readonly': None}) self.assertFalse(elem.is_editable()) @mock.patch('qutebrowser.browser.webelem.config', new=stubs.ConfigStub( {'input': {'insert-mode-on-plugins': True}})) def test_embed_true(self): """Test embed-element with insert-mode-on-plugins true.""" elem = get_webelem(tagname='embed') self.assertTrue(elem.is_editable()) @mock.patch('qutebrowser.browser.webelem.config', new=stubs.ConfigStub( {'input': {'insert-mode-on-plugins': True}})) def test_applet_true(self): """Test applet-element with insert-mode-on-plugins true.""" elem = get_webelem(tagname='applet') self.assertTrue(elem.is_editable()) @mock.patch('qutebrowser.browser.webelem.config', new=stubs.ConfigStub( {'input': {'insert-mode-on-plugins': False}})) def test_embed_false(self): """Test embed-element with insert-mode-on-plugins false.""" elem = get_webelem(tagname='embed') self.assertFalse(elem.is_editable()) @mock.patch('qutebrowser.browser.webelem.config', new=stubs.ConfigStub( {'input': {'insert-mode-on-plugins': False}})) def test_applet_false(self): """Test applet-element with insert-mode-on-plugins false.""" elem = get_webelem(tagname='applet') self.assertFalse(elem.is_editable()) def test_object_no_type(self): """Test object-element without type.""" elem = get_webelem(tagname='object') self.assertFalse(elem.is_editable()) def test_object_image(self): """Test object-element with image type.""" elem = get_webelem(tagname='object', attributes={'type': 'image/gif'}) self.assertFalse(elem.is_editable()) @mock.patch('qutebrowser.browser.webelem.config', new=stubs.ConfigStub( {'input': {'insert-mode-on-plugins': True}})) def test_object_application(self): """Test object-element with application type.""" elem = get_webelem(tagname='object', attributes={'type': 'application/foo'}) self.assertTrue(elem.is_editable()) @mock.patch('qutebrowser.browser.webelem.config', new=stubs.ConfigStub( {'input': {'insert-mode-on-plugins': False}})) def test_object_application_false(self): """Test object-element with application type but not ...-on-plugins.""" elem = get_webelem(tagname='object', attributes={'type': 'application/foo'}) self.assertFalse(elem.is_editable()) @mock.patch('qutebrowser.browser.webelem.config', new=stubs.ConfigStub( {'input': {'insert-mode-on-plugins': True}})) def test_object_classid(self): """Test object-element with classid.""" elem = get_webelem(tagname='object', attributes={'type': 'foo', 'classid': 'foo'}) self.assertTrue(elem.is_editable()) @mock.patch('qutebrowser.browser.webelem.config', new=stubs.ConfigStub( {'input': {'insert-mode-on-plugins': False}})) def test_object_classid_false(self): """Test object-element with classid but not insert-mode-on-plugins.""" elem = get_webelem(tagname='object', attributes={'type': 'foo', 'classid': 'foo'}) self.assertFalse(elem.is_editable()) def test_div_empty(self): """Test div-element without class.""" elem = get_webelem(tagname='div') self.assertFalse(elem.is_editable()) def test_div_noneditable(self): """Test div-element with non-editableclass.""" elem = get_webelem(tagname='div', classes='foo-kix-bar') self.assertFalse(elem.is_editable()) def test_div_xik(self): """Test div-element with xik class.""" elem = get_webelem(tagname='div', classes='foo kix-foo') self.assertTrue(elem.is_editable()) def test_div_xik_caps(self): """Test div-element with xik class in caps. This tests if classes are case sensitive as they should. """ elem = get_webelem(tagname='div', classes='KIX-FOO') self.assertFalse(elem.is_editable()) def test_div_codemirror(self): """Test div-element with codemirror class.""" elem = get_webelem(tagname='div', classes='foo CodeMirror-foo') self.assertTrue(elem.is_editable()) if __name__ == '__main__': unittest.main()