Implement :insert-text for QtWebEngine
This commit is contained in:
parent
ee5a972069
commit
948fa033c7
@ -1525,32 +1525,28 @@ class CommandDispatcher:
|
|||||||
self.insert_text(utils.get_clipboard())
|
self.insert_text(utils.get_clipboard())
|
||||||
|
|
||||||
@cmdutils.register(instance='command-dispatcher', maxsplit=0,
|
@cmdutils.register(instance='command-dispatcher', maxsplit=0,
|
||||||
scope='window', backend=usertypes.Backend.QtWebKit)
|
scope='window')
|
||||||
def insert_text(self, text):
|
def insert_text(self, text):
|
||||||
"""Insert text at cursor position.
|
"""Insert text at cursor position.
|
||||||
|
|
||||||
Args:
|
Args:
|
||||||
text: The text to insert.
|
text: The text to insert.
|
||||||
"""
|
"""
|
||||||
# FIXME:qtwebengine have a proper API for this
|
|
||||||
tab = self._current_widget()
|
tab = self._current_widget()
|
||||||
if not tab.has_js():
|
if not tab.has_js():
|
||||||
raise cmdexc.CommandError("This command needs javascript enabled.")
|
raise cmdexc.CommandError("This command needs javascript enabled.")
|
||||||
page = tab._widget.page() # pylint: disable=protected-access
|
|
||||||
try:
|
def _insert_text_cb(elem):
|
||||||
elem = webkitelem.focus_elem(page.currentFrame())
|
if elem is None:
|
||||||
except webkitelem.IsNullError:
|
message.error(self._win_id, "No element focused!")
|
||||||
raise cmdexc.CommandError("No element focused!")
|
return
|
||||||
if not elem.is_editable(strict=True):
|
try:
|
||||||
raise cmdexc.CommandError("Focused element is not editable!")
|
elem.insert_text(text)
|
||||||
log.misc.debug("Inserting text into element {}".format(
|
except webelem.Error as e:
|
||||||
elem.debug_text()))
|
message.error(self._win_id, str(e))
|
||||||
elem.run_js_async("""
|
return
|
||||||
var text = '{}';
|
|
||||||
var event = document.createEvent('TextEvent');
|
tab.elements.find_focused(_insert_text_cb)
|
||||||
event.initTextEvent('textInput', true, true, null, text);
|
|
||||||
this.dispatchEvent(event);
|
|
||||||
""".format(javascript.string_escape(text)))
|
|
||||||
|
|
||||||
@cmdutils.register(instance='command-dispatcher', scope='window',
|
@cmdutils.register(instance='command-dispatcher', scope='window',
|
||||||
hide=True)
|
hide=True)
|
||||||
|
@ -158,6 +158,10 @@ class AbstractWebElement(collections.abc.MutableMapping):
|
|||||||
# FIXME:qtwebengine what to do about use_js with WebEngine?
|
# FIXME:qtwebengine what to do about use_js with WebEngine?
|
||||||
raise NotImplementedError
|
raise NotImplementedError
|
||||||
|
|
||||||
|
def insert_text(self, text):
|
||||||
|
"""Insert the given text into the element."""
|
||||||
|
raise NotImplementedError
|
||||||
|
|
||||||
def run_js_async(self, code, callback=None):
|
def run_js_async(self, code, callback=None):
|
||||||
"""Run the given JS snippet async on the element."""
|
"""Run the given JS snippet async on the element."""
|
||||||
# FIXME:qtwebengine get rid of this?
|
# FIXME:qtwebengine get rid of this?
|
||||||
|
@ -108,6 +108,13 @@ class WebEngineElement(webelem.AbstractWebElement):
|
|||||||
js_code = javascript.assemble('webelem', 'set_text', self._id, text)
|
js_code = javascript.assemble('webelem', 'set_text', self._id, text)
|
||||||
self._tab.run_js_async(js_code)
|
self._tab.run_js_async(js_code)
|
||||||
|
|
||||||
|
def insert_text(self, text):
|
||||||
|
if not self.is_editable(strict=True):
|
||||||
|
raise webelem.Error("Element is not editable!")
|
||||||
|
log.misc.debug("Inserting text into element {!r}".format(self))
|
||||||
|
js_code = javascript.assemble('webelem', 'insert_text', self._id, text)
|
||||||
|
self._tab.run_js_async(js_code)
|
||||||
|
|
||||||
def run_js_async(self, code, callback=None):
|
def run_js_async(self, code, callback=None):
|
||||||
"""Run the given JS snippet async on the element."""
|
"""Run the given JS snippet async on the element."""
|
||||||
# FIXME:qtwebengine get rid of this?
|
# FIXME:qtwebengine get rid of this?
|
||||||
|
@ -135,6 +135,17 @@ class WebKitElement(webelem.AbstractWebElement):
|
|||||||
text = javascript.string_escape(text)
|
text = javascript.string_escape(text)
|
||||||
self._elem.evaluateJavaScript("this.value='{}'".format(text))
|
self._elem.evaluateJavaScript("this.value='{}'".format(text))
|
||||||
|
|
||||||
|
def insert_text(self, text):
|
||||||
|
if not self.is_editable(strict=True):
|
||||||
|
raise webelem.Error("Element is not editable!")
|
||||||
|
log.misc.debug("Inserting text into element {!r}".format(self))
|
||||||
|
self.run_js_async("""
|
||||||
|
var text = '{}';
|
||||||
|
var event = document.createEvent('TextEvent');
|
||||||
|
event.initTextEvent('textInput', true, true, null, text);
|
||||||
|
this.dispatchEvent(event);
|
||||||
|
""".format(javascript.string_escape(text)))
|
||||||
|
|
||||||
def run_js_async(self, code, callback=None):
|
def run_js_async(self, code, callback=None):
|
||||||
"""Run the given JS snippet async on the element."""
|
"""Run the given JS snippet async on the element."""
|
||||||
self._check_vanished()
|
self._check_vanished()
|
||||||
|
@ -132,6 +132,13 @@ window._qutebrowser.webelem = (function() {
|
|||||||
elements[id].value = text;
|
elements[id].value = text;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
funcs.insert_text = function(id, text) {
|
||||||
|
var elem = elements[id];
|
||||||
|
var event = document.createEvent("TextEvent");
|
||||||
|
event.initTextEvent("textInput", true, true, null, text);
|
||||||
|
elem.dispatchEvent(event);
|
||||||
|
};
|
||||||
|
|
||||||
funcs.element_at_pos = function(x, y) {
|
funcs.element_at_pos = function(x, y) {
|
||||||
// FIXME:qtwebengine
|
// FIXME:qtwebengine
|
||||||
// If the element at the specified point belongs to another document
|
// If the element at the specified point belongs to another document
|
||||||
|
@ -229,7 +229,6 @@ Feature: Yanking and pasting.
|
|||||||
|
|
||||||
#### :insert-text
|
#### :insert-text
|
||||||
|
|
||||||
@qtwebengine_todo: :insert-text is not implemented yet
|
|
||||||
Scenario: Inserting text into an empty text field
|
Scenario: Inserting text into an empty text field
|
||||||
When I open data/paste_primary.html
|
When I open data/paste_primary.html
|
||||||
And I run :click-element id qute-textarea
|
And I run :click-element id qute-textarea
|
||||||
@ -238,7 +237,6 @@ Feature: Yanking and pasting.
|
|||||||
# Compare
|
# Compare
|
||||||
Then the text field should contain "Hello world"
|
Then the text field should contain "Hello world"
|
||||||
|
|
||||||
@qtwebengine_todo: :insert-text is not implemented yet
|
|
||||||
Scenario: Inserting text into a text field at specific position
|
Scenario: Inserting text into a text field at specific position
|
||||||
When I open data/paste_primary.html
|
When I open data/paste_primary.html
|
||||||
And I set the text field to "one two three four"
|
And I set the text field to "one two three four"
|
||||||
@ -252,7 +250,6 @@ Feature: Yanking and pasting.
|
|||||||
# Compare
|
# Compare
|
||||||
Then the text field should contain "onHello worlde two three four"
|
Then the text field should contain "onHello worlde two three four"
|
||||||
|
|
||||||
@qtwebengine_todo: :insert-text is not implemented yet
|
|
||||||
Scenario: Inserting text into a text field with undo
|
Scenario: Inserting text into a text field with undo
|
||||||
When I open data/paste_primary.html
|
When I open data/paste_primary.html
|
||||||
And I run :click-element id qute-textarea
|
And I run :click-element id qute-textarea
|
||||||
@ -265,18 +262,16 @@ Feature: Yanking and pasting.
|
|||||||
# Compare
|
# Compare
|
||||||
Then the text field should contain "This text should stay"
|
Then the text field should contain "This text should stay"
|
||||||
|
|
||||||
@qtwebengine_todo: :insert-text is not implemented yet
|
|
||||||
Scenario: Inserting text without a focused field
|
Scenario: Inserting text without a focused field
|
||||||
When I open data/paste_primary.html
|
When I open data/paste_primary.html
|
||||||
And I run :enter-mode insert
|
And I run :enter-mode insert
|
||||||
And I run :insert-text test
|
And I run :insert-text test
|
||||||
Then the error "No element focused!" should be shown
|
Then the error "No element focused!" should be shown
|
||||||
|
|
||||||
@qtwebengine_todo: :insert-text is not implemented yet
|
|
||||||
Scenario: Inserting text with a read-only field
|
Scenario: Inserting text with a read-only field
|
||||||
When I open data/paste_primary.html
|
When I open data/paste_primary.html
|
||||||
And I run :click-element id qute-textarea-noedit
|
And I run :click-element id qute-textarea-noedit
|
||||||
And I wait for "Clicked non-editable element!" in the log
|
And I wait for "Clicked non-editable element!" in the log
|
||||||
And I run :enter-mode insert
|
And I run :enter-mode insert
|
||||||
And I run :insert-text test
|
And I run :insert-text test
|
||||||
Then the error "Focused element is not editable!" should be shown
|
Then the error "Element is not editable!" should be shown
|
||||||
|
@ -49,8 +49,11 @@ def test_insert_mode(file_name, elem_id, source, input_text, auto_insert,
|
|||||||
quteproc.press_keys(input_text)
|
quteproc.press_keys(input_text)
|
||||||
elif source == 'clipboard':
|
elif source == 'clipboard':
|
||||||
if request.config.webengine:
|
if request.config.webengine:
|
||||||
pytest.xfail(reason="QtWebEngine TODO: :insert-text is not "
|
pytest.xfail(reason="QtWebEngine TODO: caret mode is not "
|
||||||
"implemented")
|
"implemented")
|
||||||
|
# Note we actually run the keypress tests with QtWebEngine, as for
|
||||||
|
# some reason it selects all the text when clicking the field the
|
||||||
|
# second time.
|
||||||
quteproc.send_cmd(':debug-set-fake-clipboard "{}"'.format(input_text))
|
quteproc.send_cmd(':debug-set-fake-clipboard "{}"'.format(input_text))
|
||||||
quteproc.send_cmd(':insert-text {clipboard}')
|
quteproc.send_cmd(':insert-text {clipboard}')
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user