Merge branch 'rcorre-edit-url'
This commit is contained in:
commit
f0b66130d6
@ -14,6 +14,14 @@ This project adheres to http://semver.org/[Semantic Versioning].
|
|||||||
// `Fixed` for any bug fixes.
|
// `Fixed` for any bug fixes.
|
||||||
// `Security` to invite users to upgrade in case of vulnerabilities.
|
// `Security` to invite users to upgrade in case of vulnerabilities.
|
||||||
|
|
||||||
|
v0.7.0 (unreleased)
|
||||||
|
-------------------
|
||||||
|
|
||||||
|
Added
|
||||||
|
~~~~~
|
||||||
|
|
||||||
|
- New `:edit-url` command to edit the URL in an external editor.
|
||||||
|
|
||||||
v0.6.0
|
v0.6.0
|
||||||
------
|
------
|
||||||
|
|
||||||
|
@ -183,6 +183,7 @@ Contributors, sorted by the number of commits in descending order:
|
|||||||
* Zach-Button
|
* Zach-Button
|
||||||
* Halfwit
|
* Halfwit
|
||||||
* rikn00
|
* rikn00
|
||||||
|
* Ryan Roden-Corrent
|
||||||
* Michael Ilsaas
|
* Michael Ilsaas
|
||||||
* Martin Zimmermann
|
* Martin Zimmermann
|
||||||
* Brian Jackson
|
* Brian Jackson
|
||||||
@ -195,7 +196,6 @@ Contributors, sorted by the number of commits in descending order:
|
|||||||
* Larry Hynes
|
* Larry Hynes
|
||||||
* Johannes Altmanninger
|
* Johannes Altmanninger
|
||||||
* Samir Benmendil
|
* Samir Benmendil
|
||||||
* Ryan Roden-Corrent
|
|
||||||
* Regina Hug
|
* Regina Hug
|
||||||
* Mathias Fussenegger
|
* Mathias Fussenegger
|
||||||
* Marcelo Santos
|
* Marcelo Santos
|
||||||
|
@ -20,6 +20,7 @@
|
|||||||
|<<download-open,download-open>>|Open the last/[count]th download.
|
|<<download-open,download-open>>|Open the last/[count]th download.
|
||||||
|<<download-remove,download-remove>>|Remove the last/[count]th download from the list.
|
|<<download-remove,download-remove>>|Remove the last/[count]th download from the list.
|
||||||
|<<download-retry,download-retry>>|Retry the first failed/[count]th download.
|
|<<download-retry,download-retry>>|Retry the first failed/[count]th download.
|
||||||
|
|<<edit-url,edit-url>>|Navigate to a url formed in an external editor.
|
||||||
|<<fake-key,fake-key>>|Send a fake keypress or key string to the website or qutebrowser.
|
|<<fake-key,fake-key>>|Send a fake keypress or key string to the website or qutebrowser.
|
||||||
|<<forward,forward>>|Go forward in the history of the current tab.
|
|<<forward,forward>>|Go forward in the history of the current tab.
|
||||||
|<<fullscreen,fullscreen>>|Toggle fullscreen mode.
|
|<<fullscreen,fullscreen>>|Toggle fullscreen mode.
|
||||||
@ -223,6 +224,25 @@ Retry the first failed/[count]th download.
|
|||||||
==== count
|
==== count
|
||||||
The index of the download to retry.
|
The index of the download to retry.
|
||||||
|
|
||||||
|
[[edit-url]]
|
||||||
|
=== edit-url
|
||||||
|
Syntax: +:edit-url [*--bg*] [*--tab*] [*--window*] ['url']+
|
||||||
|
|
||||||
|
Navigate to a url formed in an external editor.
|
||||||
|
|
||||||
|
The editor which should be launched can be configured via the `general -> editor` config option.
|
||||||
|
|
||||||
|
==== positional arguments
|
||||||
|
* +'url'+: URL to edit; defaults to the current page url.
|
||||||
|
|
||||||
|
==== optional arguments
|
||||||
|
* +*-b*+, +*--bg*+: Open in a new background tab.
|
||||||
|
* +*-t*+, +*--tab*+: Open in a new tab.
|
||||||
|
* +*-w*+, +*--window*+: Open in a new window.
|
||||||
|
|
||||||
|
==== count
|
||||||
|
The tab index to open the URL in.
|
||||||
|
|
||||||
[[fake-key]]
|
[[fake-key]]
|
||||||
=== fake-key
|
=== fake-key
|
||||||
Syntax: +:fake-key [*--global*] 'keystring'+
|
Syntax: +:fake-key [*--global*] 'keystring'+
|
||||||
|
@ -65,7 +65,6 @@ class CommandDispatcher:
|
|||||||
"""
|
"""
|
||||||
|
|
||||||
def __init__(self, win_id, tabbed_browser):
|
def __init__(self, win_id, tabbed_browser):
|
||||||
self._editor = None
|
|
||||||
self._win_id = win_id
|
self._win_id = win_id
|
||||||
self._tabbed_browser = tabbed_browser
|
self._tabbed_browser = tabbed_browser
|
||||||
|
|
||||||
@ -249,7 +248,10 @@ class CommandDispatcher:
|
|||||||
try:
|
try:
|
||||||
url = urlutils.fuzzy_url(url)
|
url = urlutils.fuzzy_url(url)
|
||||||
except urlutils.InvalidUrlError as e:
|
except urlutils.InvalidUrlError as e:
|
||||||
raise cmdexc.CommandError(e)
|
# We don't use cmdexc.CommandError here as this can be called
|
||||||
|
# async from edit_url
|
||||||
|
message.error(self._win_id, str(e))
|
||||||
|
return
|
||||||
if tab or bg or window:
|
if tab or bg or window:
|
||||||
self._open(url, tab, bg, window)
|
self._open(url, tab, bg, window)
|
||||||
else:
|
else:
|
||||||
@ -1330,11 +1332,10 @@ class CommandDispatcher:
|
|||||||
text = str(elem)
|
text = str(elem)
|
||||||
else:
|
else:
|
||||||
text = elem.evaluateJavaScript('this.value')
|
text = elem.evaluateJavaScript('this.value')
|
||||||
self._editor = editor.ExternalEditor(
|
ed = editor.ExternalEditor(self._win_id, self._tabbed_browser)
|
||||||
self._win_id, self._tabbed_browser)
|
ed.editing_finished.connect(functools.partial(
|
||||||
self._editor.editing_finished.connect(
|
self.on_editing_finished, elem))
|
||||||
functools.partial(self.on_editing_finished, elem))
|
ed.edit(text)
|
||||||
self._editor.edit(text)
|
|
||||||
|
|
||||||
def on_editing_finished(self, elem, text):
|
def on_editing_finished(self, elem, text):
|
||||||
"""Write the editor text into the form field and clean up tempfile.
|
"""Write the editor text into the form field and clean up tempfile.
|
||||||
@ -1847,3 +1848,29 @@ class CommandDispatcher:
|
|||||||
"""Clear remembered SSL error answers."""
|
"""Clear remembered SSL error answers."""
|
||||||
nam = self._current_widget().page().networkAccessManager()
|
nam = self._current_widget().page().networkAccessManager()
|
||||||
nam.clear_all_ssl_errors()
|
nam.clear_all_ssl_errors()
|
||||||
|
|
||||||
|
@cmdutils.register(instance='command-dispatcher', scope='window',
|
||||||
|
count='count')
|
||||||
|
def edit_url(self, url=None, bg=False, tab=False, window=False,
|
||||||
|
count=None):
|
||||||
|
"""Navigate to a url formed in an external editor.
|
||||||
|
|
||||||
|
The editor which should be launched can be configured via the
|
||||||
|
`general -> editor` config option.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
url: URL to edit; defaults to the current page url.
|
||||||
|
bg: Open in a new background tab.
|
||||||
|
tab: Open in a new tab.
|
||||||
|
window: Open in a new window.
|
||||||
|
count: The tab index to open the URL in, or None.
|
||||||
|
"""
|
||||||
|
cmdutils.check_exclusive((tab, bg, window), 'tbw')
|
||||||
|
|
||||||
|
ed = editor.ExternalEditor(self._win_id, self._tabbed_browser)
|
||||||
|
|
||||||
|
# Passthrough for openurl args (e.g. -t, -b, -w)
|
||||||
|
ed.editing_finished.connect(functools.partial(
|
||||||
|
self.openurl, bg=bg, tab=tab, window=window, count=count))
|
||||||
|
|
||||||
|
ed.edit(url or self._current_url().toString())
|
||||||
|
72
tests/integration/features/editor.feature
Normal file
72
tests/integration/features/editor.feature
Normal file
@ -0,0 +1,72 @@
|
|||||||
|
Feature: Opening external editors
|
||||||
|
|
||||||
|
## :edit-url
|
||||||
|
|
||||||
|
Scenario: Editing an URL
|
||||||
|
When I open data/numbers/1.txt
|
||||||
|
And I set up a fake editor replacing "1.txt" by "2.txt"
|
||||||
|
And I run :edit-url
|
||||||
|
Then data/numbers/2.txt should be loaded
|
||||||
|
|
||||||
|
Scenario: Editing an URL with -t
|
||||||
|
When I run :tab-only
|
||||||
|
And I open data/numbers/1.txt
|
||||||
|
And I set up a fake editor replacing "1.txt" by "2.txt"
|
||||||
|
And I run :edit-url -t
|
||||||
|
Then data/numbers/2.txt should be loaded
|
||||||
|
And the following tabs should be open:
|
||||||
|
- data/numbers/1.txt
|
||||||
|
- data/numbers/2.txt (active)
|
||||||
|
|
||||||
|
Scenario: Editing an URL with -b
|
||||||
|
When I run :tab-only
|
||||||
|
And I open data/numbers/1.txt
|
||||||
|
And I set up a fake editor replacing "1.txt" by "2.txt"
|
||||||
|
And I run :edit-url -b
|
||||||
|
Then data/numbers/2.txt should be loaded
|
||||||
|
And the following tabs should be open:
|
||||||
|
- data/numbers/1.txt (active)
|
||||||
|
- data/numbers/2.txt
|
||||||
|
|
||||||
|
Scenario: Editing an URL with -w
|
||||||
|
When I open data/numbers/1.txt in a new tab
|
||||||
|
And I run :tab-only
|
||||||
|
And I set up a fake editor replacing "1.txt" by "2.txt"
|
||||||
|
And I run :edit-url -w
|
||||||
|
Then data/numbers/2.txt should be loaded
|
||||||
|
And the session should look like:
|
||||||
|
windows:
|
||||||
|
- tabs:
|
||||||
|
- active: true
|
||||||
|
history:
|
||||||
|
- active: true
|
||||||
|
url: http://localhost:*/data/numbers/1.txt
|
||||||
|
- tabs:
|
||||||
|
- active: true
|
||||||
|
history:
|
||||||
|
- active: true
|
||||||
|
url: http://localhost:*/data/numbers/2.txt
|
||||||
|
|
||||||
|
Scenario: Editing an URL with count
|
||||||
|
Given I have a fresh instance
|
||||||
|
When I open data/numbers/1.txt
|
||||||
|
And I run :tab-only
|
||||||
|
And I open about:blank in a new tab
|
||||||
|
And I run :tab-focus 1
|
||||||
|
And I set up a fake editor replacing "1.txt" by "2.txt"
|
||||||
|
And I run :edit-url with count 2
|
||||||
|
Then data/numbers/2.txt should be loaded
|
||||||
|
And the following tabs should be open:
|
||||||
|
- data/numbers/1.txt (active)
|
||||||
|
- data/numbers/2.txt
|
||||||
|
|
||||||
|
Scenario: Editing an URL with -t and -b
|
||||||
|
When I run :edit-url -t -b
|
||||||
|
Then the error "Only one of -t/-b/-w can be given!" should be shown
|
||||||
|
|
||||||
|
Scenario: Editing an URL with invalid URL
|
||||||
|
When I set general -> auto-search to false
|
||||||
|
And I open data/hello.txt
|
||||||
|
And I set up a fake editor replacing "http://localhost:(port)/data/hello.txt" by "foo!"
|
||||||
|
And I run :edit-url
|
||||||
|
Then the error "Invalid URL" should be shown
|
45
tests/integration/features/test_editor.py
Normal file
45
tests/integration/features/test_editor.py
Normal file
@ -0,0 +1,45 @@
|
|||||||
|
# vim: ft=python fileencoding=utf-8 sts=4 sw=4 et:
|
||||||
|
|
||||||
|
# Copyright 2016 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/>.
|
||||||
|
|
||||||
|
import sys
|
||||||
|
import textwrap
|
||||||
|
|
||||||
|
import pytest_bdd as bdd
|
||||||
|
bdd.scenarios('editor.feature')
|
||||||
|
|
||||||
|
|
||||||
|
@bdd.when(bdd.parsers.parse('I set up a fake editor replacing "{text}" by '
|
||||||
|
'"{replacement}"'))
|
||||||
|
def set_up_editor_replacement(quteproc, httpbin, tmpdir, text, replacement):
|
||||||
|
"""Set up general -> editor to a small python script doing a replacement"""
|
||||||
|
text = text.replace('(port)', str(httpbin.port))
|
||||||
|
script = tmpdir / 'script.py'
|
||||||
|
script.write(textwrap.dedent("""
|
||||||
|
import sys
|
||||||
|
|
||||||
|
with open(sys.argv[1], encoding='utf-8') as f:
|
||||||
|
data = f.read()
|
||||||
|
|
||||||
|
data = data.replace("{text}", "{replacement}")
|
||||||
|
|
||||||
|
with open(sys.argv[1], 'w', encoding='utf-8') as f:
|
||||||
|
f.write(data)
|
||||||
|
""".format(text=text, replacement=replacement)))
|
||||||
|
editor = '"{}" "{}" {{}}'.format(sys.executable, script)
|
||||||
|
quteproc.set_setting('general', 'editor', editor)
|
Loading…
Reference in New Issue
Block a user