Merge _split into _partition in Completer.

After the refactoring, _split is only called by _partition so just make
it part of the same method. This also removes the use of
_empty_item_index, as it can be figured out on the fly.
This commit is contained in:
Ryan Roden-Corrent 2016-09-11 07:56:42 -04:00
parent 127412d91c
commit fcadde6aef

View File

@ -46,7 +46,6 @@ class Completer(QObject):
self._win_id = win_id self._win_id = win_id
self._cmd = cmd self._cmd = cmd
self._ignore_change = False self._ignore_change = False
self._empty_item_idx = None
self._timer = QTimer() self._timer = QTimer()
self._timer.setSingleShot(True) self._timer.setSingleShot(True)
self._timer.setInterval(0) self._timer.setInterval(0)
@ -150,7 +149,18 @@ class Completer(QObject):
return s return s
def _partition(self): def _partition(self):
parts = [x for x in self._split(keep=True) if x] """Divide the commandline text into chunks around the cursor position.
Return:
([parts_before_cursor], 'part_under_cursor', [parts_after_cursor])
"""
text = self._cmd.text()[len(self._cmd.prefix()):]
if not text or not text.strip():
# Only ":", empty part under the cursor with nothing before/after
return [], '', []
runner = runners.CommandRunner(self._win_id)
result = runner.parse(text, fallback=True, keep=True)
parts = [x for x in result.cmdline if x]
pos = self._cmd.cursorPosition() - len(self._cmd.prefix()) pos = self._cmd.cursorPosition() - len(self._cmd.prefix())
log.completion.debug('partitioning {} around position {}'.format(parts, log.completion.debug('partitioning {} around position {}'.format(parts,
pos)) pos))
@ -165,10 +175,8 @@ class Completer(QObject):
# strip trailing whitepsace included as a separate token # strip trailing whitepsace included as a separate token
postfix = [x.strip() for x in parts[i+1:] if not x.isspace()] postfix = [x.strip() for x in parts[i+1:] if not x.isspace()]
log.completion.debug( log.completion.debug(
'partitioned: {} {} {}'.format(prefix, center, postfix)) "partitioned: {} '{}' {}".format(prefix, center, postfix))
return prefix, center, postfix return prefix, center, postfix
log.completion.debug('empty partition result')
return [], '', []
@pyqtSlot(QItemSelection) @pyqtSlot(QItemSelection)
def on_selection_changed(self, selected): def on_selection_changed(self, selected):
@ -187,21 +195,22 @@ class Completer(QObject):
data = model.data(indexes[0]) data = model.data(indexes[0])
if data is None: if data is None:
return return
parts = self._split() before, center, after = self._partition()
log.completion.debug("Changing {} to '{}'".format(center, data))
try: try:
needs_quoting = cmdutils.cmd_dict[parts[0]].maxsplit is None needs_quoting = cmdutils.cmd_dict[before[0]].maxsplit is None
except KeyError: except (KeyError, IndexError):
needs_quoting = True needs_quoting = True
if needs_quoting: if needs_quoting:
data = self._quote(data) data = self._quote(data)
if model.count() == 1 and config.get('completion', 'quick-complete'): if model.count() == 1 and config.get('completion', 'quick-complete'):
# If we only have one item, we want to apply it immediately # If we only have one item, we want to apply it immediately
# and go on to the next part. # and go on to the next part.
self._change_completed_part(data, immediate=True) self._change_completed_part(data, before, after, immediate=True)
else: else:
log.completion.debug("Will ignore next completion update.") log.completion.debug("Will ignore next completion update.")
self._ignore_change = True self._ignore_change = True
self._change_completed_part(data) self._change_completed_part(data, before, after)
@pyqtSlot() @pyqtSlot()
def schedule_completion_update(self): def schedule_completion_update(self):
@ -252,43 +261,17 @@ class Completer(QObject):
completion.set_model(model, pattern) completion.set_model(model, pattern)
def _split(self, keep=False): def _change_completed_part(self, newtext, before, after, immediate=False):
"""Get the text split up in parts.
Args:
keep: Whether to keep special chars and whitespace.
aliases: Whether to resolve aliases.
"""
text = self._cmd.text()[len(self._cmd.prefix()):]
if not text:
# When only ":" is entered, we already have one imaginary part,
# which just is empty at the moment.
return ['']
if not text.strip():
# Text is only whitespace so we treat this as a single element with
# the whitespace.
return [text]
runner = runners.CommandRunner(self._win_id)
result = runner.parse(text, fallback=True, keep=keep)
parts = result.cmdline
if self._empty_item_idx is not None:
log.completion.debug("Empty element queued at {}, "
"inserting.".format(self._empty_item_idx))
parts.insert(self._empty_item_idx, '')
#log.completion.debug("Splitting '{}' -> {}".format(text, parts))
return parts
def _change_completed_part(self, newtext, immediate=False):
"""Change the part we're currently completing in the commandline. """Change the part we're currently completing in the commandline.
Args: Args:
text: The text to set (string). text: The text to set (string) for the token under the cursor.
before: Commandline tokens before the token under the cursor.
after: Commandline tokens after the token under the cursor.
immediate: True if the text should be completed immediately immediate: True if the text should be completed immediately
including a trailing space and we shouldn't continue including a trailing space and we shouldn't continue
completing the current item. completing the current item.
""" """
before, center, after = self._partition()
log.completion.debug("changing {} to '{}'".format(center, newtext))
text = self._cmd.prefix() + ' '.join([*before, newtext]) text = self._cmd.prefix() + ' '.join([*before, newtext])
pos = len(text) + (1 if immediate else 0) pos = len(text) + (1 if immediate else 0)
if after: if after: