Selection implentation

This commit is contained in:
Artur Shaikhullin 2017-11-22 19:56:05 +06:00
parent 8aca37e5d5
commit e7a66d92a8
5 changed files with 64 additions and 26 deletions

View File

@ -392,7 +392,7 @@ class AbstractCaret(QObject):
def has_selection(self): def has_selection(self):
raise NotImplementedError raise NotImplementedError
def selection(self, html=False): def selection(self, html=False, selection=None):
raise NotImplementedError raise NotImplementedError
def follow_selected(self, *, tab=False): def follow_selected(self, *, tab=False):

View File

@ -838,6 +838,11 @@ class CommandDispatcher:
sel: Use the primary selection instead of the clipboard. sel: Use the primary selection instead of the clipboard.
keep: Stay in visual mode after yanking the selection. keep: Stay in visual mode after yanking the selection.
""" """
self.yank_object = {
'sel': sel,
'what': what,
'keep': keep
}
if what == 'title': if what == 'title':
s = self._tabbed_browser.page_title(self._current_index()) s = self._tabbed_browser.page_title(self._current_index())
elif what == 'domain': elif what == 'domain':
@ -850,28 +855,36 @@ class CommandDispatcher:
what = 'URL' # For printing what = 'URL' # For printing
elif what == 'selection': elif what == 'selection':
caret = self._current_widget().caret caret = self._current_widget().caret
s = caret.selection() caret.selection(False, self._yank_callback)
if not caret.has_selection() or not s:
message.info("Nothing to yank")
return return
else: # pragma: no cover else: # pragma: no cover
raise ValueError("Invalid value {!r} for `what'.".format(what)) raise ValueError("Invalid value {!r} for `what'.".format(what))
if sel and utils.supports_selection(): self._yank_to_target(s)
def _yank_callback(self, s):
if not self._current_widget().caret.has_selection() or not s:
message.info("Nothing to yank")
return
self._yank_to_target(s)
def _yank_to_target(self, s):
if self.yank_object['sel'] and utils.supports_selection():
target = "primary selection" target = "primary selection"
else: else:
sel = False self.yank_object['sel'] = False
target = "clipboard" target = "clipboard"
utils.set_clipboard(s, selection=sel) utils.set_clipboard(s, selection=self.yank_object['sel'])
if what != 'selection': if self.yank_object['what'] != 'selection':
message.info("Yanked {} to {}: {}".format(what, target, s)) message.info("Yanked {} to {}: {}".format(self.yank_object['what'], target, s))
else: else:
message.info("{} {} yanked to {}".format( message.info("{} {} yanked to {}".format(
len(s), "char" if len(s) == 1 else "chars", target)) len(s), "char" if len(s) == 1 else "chars", target))
if not keep: if not self.yank_object['keep']:
modeman.leave(self._win_id, KeyMode.caret, "yank selected", modeman.leave(self._win_id, KeyMode.caret, "yank selected",
maybe=True) maybe=True)
self.yank_object = None
@cmdutils.register(instance='command-dispatcher', scope='window') @cmdutils.register(instance='command-dispatcher', scope='window')
@cmdutils.argument('count', count=True) @cmdutils.argument('count', count=True)

View File

@ -231,18 +231,22 @@ class WebEngineCaret(browsertab.AbstractCaret):
javascript.assemble('caret', 'moveToEndOfLine')) javascript.assemble('caret', 'moveToEndOfLine'))
def move_to_start_of_next_block(self, count=1): def move_to_start_of_next_block(self, count=1):
for _ in range(count):
self._tab.run_js_async( self._tab.run_js_async(
javascript.assemble('caret', 'moveToStartOfNextBlock')) javascript.assemble('caret', 'moveToStartOfNextBlock'))
def move_to_start_of_prev_block(self, count=1): def move_to_start_of_prev_block(self, count=1):
for _ in range(count):
self._tab.run_js_async( self._tab.run_js_async(
javascript.assemble('caret', 'moveToStartOfPrevBlock')) javascript.assemble('caret', 'moveToStartOfPrevBlock'))
def move_to_end_of_next_block(self, count=1): def move_to_end_of_next_block(self, count=1):
for _ in range(count):
self._tab.run_js_async( self._tab.run_js_async(
javascript.assemble('caret', 'moveToEndOfNextBlock')) javascript.assemble('caret', 'moveToEndOfNextBlock'))
def move_to_end_of_prev_block(self, count=1): def move_to_end_of_prev_block(self, count=1):
for _ in range(count):
self._tab.run_js_async( self._tab.run_js_async(
javascript.assemble('caret', 'moveToEndOfPrevBlock')) javascript.assemble('caret', 'moveToEndOfPrevBlock'))
@ -259,15 +263,28 @@ class WebEngineCaret(browsertab.AbstractCaret):
javascript.assemble('caret', 'toggleSelection')) javascript.assemble('caret', 'toggleSelection'))
def drop_selection(self): def drop_selection(self):
log.stub() self._tab.run_js_async(
javascript.assemble('caret', 'dropSelection'))
def has_selection(self): def has_selection(self):
if qtutils.version_check('5.10'):
return self._widget.hasSelection() return self._widget.hasSelection()
else:
# WORKAROUND for
# https://bugreports.qt.io/browse/QTBUG-53134
return True
def selection(self, html=False): def selection(self, html=False, callback=None):
if html: if html:
raise browsertab.UnsupportedOperationError raise browsertab.UnsupportedOperationError
return self._widget.selectedText() if qtutils.version_check('5.10'):
callback(self._widget.selectedText())
else:
# WORKAROUND for
# https://bugreports.qt.io/browse/QTBUG-53134
self._tab.run_js_async(
'window.getSelection().toString()', callback)
self.drop_selection()
def _follow_selected_cb(self, js_elem, tab=False): def _follow_selected_cb(self, js_elem, tab=False):
"""Callback for javascript which clicks the selected element. """Callback for javascript which clicks the selected element.

View File

@ -335,10 +335,10 @@ class WebKitCaret(browsertab.AbstractCaret):
def has_selection(self): def has_selection(self):
return self._widget.hasSelection() return self._widget.hasSelection()
def selection(self, html=False): def selection(self, html=False, callback=False):
if html: if html:
return self._widget.selectedHtml() callback(self._widget.selectedHtml())
return self._widget.selectedText() callback(self._widget.selectedText())
def follow_selected(self, *, tab=False): def follow_selected(self, *, tab=False):
if not self.has_selection(): if not self.has_selection():

View File

@ -1690,6 +1690,14 @@ window._qutebrowser.caret = (function() {
CaretBrowsing.move('forward', 'documentboundary'); CaretBrowsing.move('forward', 'documentboundary');
} }
funcs.dropSelection = () => {
window.getSelection().removeAllRanges();
}
funcs.getSelection = () => {
return window.getSelection().toString();
}
funcs.toggleSelection = () => { funcs.toggleSelection = () => {
CaretBrowsing.selectionEnabled = !CaretBrowsing.selectionEnabled; CaretBrowsing.selectionEnabled = !CaretBrowsing.selectionEnabled;
} }