/vim_runtime/sources_non_forked/jedi-vim/pythonx/jedi_vim.py
Python | 1159 lines | 955 code | 106 blank | 98 comment | 146 complexity | ceb541bd30a7c1e36b936c542ea2e8ed MD5 | raw file
- # -*- coding: utf-8 -*-
- """
- The Python parts of the Jedi library for VIM. It is mostly about communicating
- with VIM.
- """
- import traceback # for exception output
- import re
- import os
- import sys
- from shlex import split as shsplit
- from contextlib import contextmanager
- try:
- from itertools import zip_longest
- except ImportError:
- from itertools import izip_longest as zip_longest # Python 2
- import vim
- is_py3 = sys.version_info[0] >= 3
- if is_py3:
- ELLIPSIS = "…"
- unicode = str
- else:
- ELLIPSIS = u"…"
- try:
- # Somehow sys.prefix is set in combination with VIM and virtualenvs.
- # However the sys path is not affected. Just reset it to the normal value.
- sys.prefix = sys.base_prefix
- sys.exec_prefix = sys.base_exec_prefix
- except AttributeError:
- # If we're not in a virtualenv we don't care. Everything is fine.
- pass
- class PythonToVimStr(unicode):
- """ Vim has a different string implementation of single quotes """
- __slots__ = []
- def __new__(cls, obj, encoding='UTF-8'):
- if not (is_py3 or isinstance(obj, unicode)):
- obj = unicode.__new__(cls, obj, encoding)
- # Vim cannot deal with zero bytes:
- obj = obj.replace('\0', '\\0')
- return unicode.__new__(cls, obj)
- def __repr__(self):
- # this is totally stupid and makes no sense but vim/python unicode
- # support is pretty bad. don't ask how I came up with this... It just
- # works...
- # It seems to be related to that bug: http://bugs.python.org/issue5876
- if unicode is str:
- s = self
- else:
- s = self.encode('UTF-8')
- return '"%s"' % s.replace('\\', '\\\\').replace('"', r'\"')
- class VimError(Exception):
- def __init__(self, message, throwpoint, executing):
- super(type(self), self).__init__(message)
- self.message = message
- self.throwpoint = throwpoint
- self.executing = executing
- def __str__(self):
- return "{}; created by {!r} (in {})".format(
- self.message, self.executing, self.throwpoint
- )
- def _catch_exception(string, is_eval):
- """
- Interface between vim and python calls back to it.
- Necessary, because the exact error message is not given by `vim.error`.
- """
- result = vim.eval('jedi#_vim_exceptions({0}, {1})'.format(
- repr(PythonToVimStr(string, 'UTF-8')), int(is_eval)))
- if 'exception' in result:
- raise VimError(result['exception'], result['throwpoint'], string)
- return result['result']
- def vim_command(string):
- _catch_exception(string, is_eval=False)
- def vim_eval(string):
- return _catch_exception(string, is_eval=True)
- def no_jedi_warning(error=None):
- vim.command('echohl WarningMsg')
- vim.command('echom "Please install Jedi if you want to use jedi-vim."')
- if error:
- vim.command('echom "The error was: {0}"'.format(error))
- vim.command('echohl None')
- def echo_highlight(msg):
- vim_command('echohl WarningMsg | echom "jedi-vim: {0}" | echohl None'.format(
- str(msg).replace('"', '\\"')))
- jedi_path = os.path.join(os.path.dirname(__file__), 'jedi')
- sys.path.insert(0, jedi_path)
- parso_path = os.path.join(os.path.dirname(__file__), 'parso')
- sys.path.insert(0, parso_path)
- try:
- import jedi
- except ImportError:
- jedi = None
- jedi_import_error = sys.exc_info()
- else:
- try:
- version = jedi.__version__
- except Exception as e: # e.g. AttributeError
- echo_highlight(
- "Error when loading the jedi python module ({0}). "
- "Please ensure that Jedi is installed correctly (see Installation "
- "in the README.".format(e))
- jedi = None
- else:
- if isinstance(version, str):
- # the normal use case, now.
- from jedi import utils
- version = utils.version_info()
- if version < (0, 7):
- echo_highlight('Please update your Jedi version, it is too old.')
- finally:
- sys.path.remove(jedi_path)
- sys.path.remove(parso_path)
- class VimCompat:
- _eval_cache = {}
- _func_cache = {}
- @classmethod
- def has(cls, what):
- try:
- return cls._eval_cache[what]
- except KeyError:
- ret = cls._eval_cache[what] = cls.call('has', what)
- return ret
- @classmethod
- def call(cls, func, *args):
- try:
- f = cls._func_cache[func]
- except KeyError:
- if IS_NVIM:
- f = cls._func_cache[func] = getattr(vim.funcs, func)
- else:
- f = cls._func_cache[func] = vim.Function(func)
- return f(*args)
- @classmethod
- def setqflist(cls, items, title, context):
- if cls.has('patch-7.4.2200'): # can set qf title.
- what = {'title': title}
- if cls.has('patch-8.0.0590'): # can set qf context
- what['context'] = {'jedi_usages': context}
- if cls.has('patch-8.0.0657'): # can set items via "what".
- what['items'] = items
- cls.call('setqflist', [], ' ', what)
- else:
- # Can set title (and maybe context), but needs two calls.
- cls.call('setqflist', items)
- cls.call('setqflist', items, 'a', what)
- else:
- cls.call('setqflist', items)
- @classmethod
- def setqflist_title(cls, title):
- if cls.has('patch-7.4.2200'):
- cls.call('setqflist', [], 'a', {'title': title})
- @classmethod
- def can_update_current_qflist_for_context(cls, context):
- if cls.has('patch-8.0.0590'): # can set qf context
- return cls.call('getqflist', {'context': 1})['context'] == {
- 'jedi_usages': context,
- }
- def catch_and_print_exceptions(func):
- def wrapper(*args, **kwargs):
- try:
- return func(*args, **kwargs)
- except (Exception, vim.error):
- print(traceback.format_exc())
- return None
- return wrapper
- def _check_jedi_availability(show_error=False):
- def func_receiver(func):
- def wrapper(*args, **kwargs):
- if jedi is None:
- if show_error:
- no_jedi_warning()
- return
- else:
- return func(*args, **kwargs)
- return wrapper
- return func_receiver
- current_environment = (None, None)
- def get_environment(use_cache=True):
- global current_environment
- vim_force_python_version = vim_eval("g:jedi#force_py_version")
- if use_cache and vim_force_python_version == current_environment[0]:
- return current_environment[1]
- environment = None
- if vim_force_python_version == "auto":
- environment = jedi.api.environment.get_cached_default_environment()
- else:
- force_python_version = vim_force_python_version
- if '0000' in force_python_version or '9999' in force_python_version:
- # It's probably a float that wasn't shortened.
- try:
- force_python_version = "{:.1f}".format(float(force_python_version))
- except ValueError:
- pass
- elif isinstance(force_python_version, float):
- force_python_version = "{:.1f}".format(force_python_version)
- try:
- environment = jedi.get_system_environment(force_python_version)
- except jedi.InvalidPythonEnvironment as exc:
- environment = jedi.api.environment.get_cached_default_environment()
- echo_highlight(
- "force_python_version=%s is not supported: %s - using %s." % (
- vim_force_python_version, str(exc), str(environment)))
- current_environment = (vim_force_python_version, environment)
- return environment
- def get_known_environments():
- """Get known Jedi environments."""
- envs = list(jedi.api.environment.find_virtualenvs())
- envs.extend(jedi.api.environment.find_system_environments())
- return envs
- @catch_and_print_exceptions
- def get_script(source=None, column=None):
- jedi.settings.additional_dynamic_modules = [
- b.name for b in vim.buffers if (
- b.name is not None and
- b.name.endswith('.py') and
- b.options['buflisted'])]
- if source is None:
- source = '\n'.join(vim.current.buffer)
- row = vim.current.window.cursor[0]
- if column is None:
- column = vim.current.window.cursor[1]
- buf_path = vim.current.buffer.name
- return jedi.Script(
- source, row, column, buf_path,
- encoding=vim_eval('&encoding') or 'latin1',
- environment=get_environment(),
- )
- @_check_jedi_availability(show_error=False)
- @catch_and_print_exceptions
- def completions():
- row, column = vim.current.window.cursor
- # Clear call signatures in the buffer so they aren't seen by the completer.
- # Call signatures in the command line can stay.
- if int(vim_eval("g:jedi#show_call_signatures")) == 1:
- clear_call_signatures()
- if vim.eval('a:findstart') == '1':
- count = 0
- for char in reversed(vim.current.line[:column]):
- if not re.match(r'[\w\d]', char):
- break
- count += 1
- vim.command('return %i' % (column - count))
- else:
- base = vim.eval('a:base')
- source = ''
- for i, line in enumerate(vim.current.buffer):
- # enter this path again, otherwise source would be incomplete
- if i == row - 1:
- source += line[:column] + base + line[column:]
- else:
- source += line
- source += '\n'
- # here again hacks, because jedi has a different interface than vim
- column += len(base)
- try:
- script = get_script(source=source, column=column)
- completions = script.completions()
- signatures = script.call_signatures()
- add_info = "preview" in vim.eval("&completeopt").split(",")
- out = []
- for c in completions:
- d = dict(word=PythonToVimStr(c.name[:len(base)] + c.complete),
- abbr=PythonToVimStr(c.name_with_symbols),
- # stuff directly behind the completion
- menu=PythonToVimStr(c.description),
- icase=1, # case insensitive
- dup=1 # allow duplicates (maybe later remove this)
- )
- if add_info:
- try:
- d["info"] = PythonToVimStr(c.docstring())
- except Exception:
- print("jedi-vim: error with docstring for %r: %s" % (
- c, traceback.format_exc()))
- out.append(d)
- strout = str(out)
- except Exception:
- # print to stdout, will be in :messages
- print(traceback.format_exc())
- strout = ''
- completions = []
- signatures = []
- show_call_signatures(signatures)
- vim.command('return ' + strout)
- @contextmanager
- def tempfile(content):
- # Using this instead of the tempfile module because Windows won't read
- # from a file not yet written to disk
- with open(vim_eval('tempname()'), 'w') as f:
- f.write(content)
- try:
- yield f
- finally:
- os.unlink(f.name)
- @_check_jedi_availability(show_error=True)
- @catch_and_print_exceptions
- def goto(mode="goto"):
- """
- :param str mode: "definition", "assignment", "goto"
- :return: list of definitions/assignments
- :rtype: list
- """
- script = get_script()
- if mode == "goto":
- definitions = script.goto_assignments(follow_imports=True)
- elif mode == "definition":
- definitions = script.goto_definitions()
- elif mode == "assignment":
- definitions = script.goto_assignments()
- elif mode == "stubs":
- definitions = script.goto_assignments(follow_imports=True, only_stubs=True)
- if not definitions:
- echo_highlight("Couldn't find any definitions for this.")
- elif len(definitions) == 1 and mode != "related_name":
- d = list(definitions)[0]
- if d.column is None:
- if d.is_keyword:
- echo_highlight("Cannot get the definition of Python keywords.")
- else:
- echo_highlight("Builtin modules cannot be displayed (%s)."
- % d.desc_with_module)
- else:
- using_tagstack = int(vim_eval('g:jedi#use_tag_stack')) == 1
- if (d.module_path or '') != vim.current.buffer.name:
- result = new_buffer(d.module_path,
- using_tagstack=using_tagstack)
- if not result:
- return []
- if (using_tagstack and d.module_path and
- os.path.exists(d.module_path)):
- tagname = d.name
- with tempfile('{0}\t{1}\t{2}'.format(
- tagname, d.module_path, 'call cursor({0}, {1})'.format(
- d.line, d.column + 1))) as f:
- old_tags = vim.eval('&tags')
- old_wildignore = vim.eval('&wildignore')
- try:
- # Clear wildignore to ensure tag file isn't ignored
- vim.command('set wildignore=')
- vim.command('let &tags = %s' %
- repr(PythonToVimStr(f.name)))
- vim.command('tjump %s' % tagname)
- finally:
- vim.command('let &tags = %s' %
- repr(PythonToVimStr(old_tags)))
- vim.command('let &wildignore = %s' %
- repr(PythonToVimStr(old_wildignore)))
- vim.current.window.cursor = d.line, d.column
- else:
- show_goto_multi_results(definitions, mode)
- return definitions
- def relpath(path):
- """Make path relative to cwd if it is below."""
- abspath = os.path.abspath(path)
- if abspath.startswith(os.getcwd()):
- return os.path.relpath(path)
- return path
- def annotate_description(d):
- code = d.get_line_code().strip()
- if d.type == 'statement':
- return code
- if d.type == 'function':
- if code.startswith('def'):
- return code
- typ = 'def'
- else:
- typ = d.type
- return '[%s] %s' % (typ, code)
- def show_goto_multi_results(definitions, mode):
- """Create (or reuse) a quickfix list for multiple definitions."""
- global _current_definitions
- lst = []
- (row, col) = vim.current.window.cursor
- current_idx = None
- current_def = None
- for d in definitions:
- if d.column is None:
- # Typically a namespace, in the future maybe other things as
- # well.
- lst.append(dict(text=PythonToVimStr(d.description)))
- else:
- text = annotate_description(d)
- lst.append(dict(filename=PythonToVimStr(relpath(d.module_path)),
- lnum=d.line, col=d.column + 1,
- text=PythonToVimStr(text)))
- # Select current/nearest entry via :cc later.
- if d.line == row and d.column <= col:
- if (current_idx is None
- or (abs(lst[current_idx]["col"] - col)
- > abs(d.column - col))):
- current_idx = len(lst)
- current_def = d
- # Build qflist title.
- qf_title = mode
- if current_def is not None:
- if current_def.full_name:
- qf_title += ": " + current_def.full_name
- else:
- qf_title += ": " + str(current_def)
- select_entry = current_idx
- else:
- select_entry = 0
- qf_context = id(definitions)
- if (_current_definitions
- and VimCompat.can_update_current_qflist_for_context(qf_context)):
- # Same list, only adjust title/selected entry.
- VimCompat.setqflist_title(qf_title)
- vim_command('%dcc' % select_entry)
- else:
- VimCompat.setqflist(lst, title=qf_title, context=qf_context)
- for_usages = mode == "usages"
- vim_eval('jedi#add_goto_window(%d, %d)' % (for_usages, len(lst)))
- vim_command('%d' % select_entry)
- def _same_definitions(a, b):
- """Compare without _inference_state.
- Ref: https://github.com/davidhalter/jedi-vim/issues/952)
- """
- return all(
- x._name.start_pos == y._name.start_pos
- and x.module_path == y.module_path
- and x.name == y.name
- for x, y in zip(a, b)
- )
- @catch_and_print_exceptions
- def usages(visuals=True):
- script = get_script()
- definitions = script.usages()
- if not definitions:
- echo_highlight("No usages found here.")
- return definitions
- if visuals:
- global _current_definitions
- if _current_definitions:
- if _same_definitions(_current_definitions, definitions):
- definitions = _current_definitions
- else:
- clear_usages()
- assert not _current_definitions
- show_goto_multi_results(definitions, "usages")
- if not _current_definitions:
- _current_definitions = definitions
- highlight_usages()
- else:
- assert definitions is _current_definitions # updated above
- return definitions
- _current_definitions = None
- """Current definitions to use for highlighting."""
- _pending_definitions = {}
- """Pending definitions for unloaded buffers."""
- _placed_definitions_in_buffers = set()
- """Set of buffers for faster cleanup."""
- IS_NVIM = hasattr(vim, 'from_nvim')
- if IS_NVIM:
- vim_prop_add = None
- else:
- vim_prop_type_added = False
- try:
- vim_prop_add = vim.Function("prop_add")
- except ValueError:
- vim_prop_add = None
- else:
- vim_prop_remove = vim.Function("prop_remove")
- def clear_usages():
- """Clear existing highlights."""
- global _current_definitions
- if _current_definitions is None:
- return
- _current_definitions = None
- if IS_NVIM:
- for buf in _placed_definitions_in_buffers:
- src_ids = buf.vars.get('_jedi_usages_src_ids')
- if src_ids is not None:
- for src_id in src_ids:
- buf.clear_highlight(src_id)
- elif vim_prop_add:
- for buf in _placed_definitions_in_buffers:
- vim_prop_remove({
- 'type': 'jediUsage',
- 'all': 1,
- 'bufnr': buf.number,
- })
- else:
- # Unset current window only.
- assert _current_definitions is None
- highlight_usages_for_vim_win()
- _placed_definitions_in_buffers.clear()
- def highlight_usages():
- """Set definitions to be highlighted.
- With Neovim it will use the nvim_buf_add_highlight API to highlight all
- buffers already.
- With Vim without support for text-properties only the current window is
- highlighted via matchaddpos, and autocommands are setup to highlight other
- windows on demand. Otherwise Vim's text-properties are used.
- """
- global _current_definitions, _pending_definitions
- definitions = _current_definitions
- _pending_definitions = {}
- if IS_NVIM or vim_prop_add:
- bufs = {x.name: x for x in vim.buffers}
- defs_per_buf = {}
- for definition in definitions:
- try:
- buf = bufs[definition.module_path]
- except KeyError:
- continue
- defs_per_buf.setdefault(buf, []).append(definition)
- if IS_NVIM:
- # We need to remember highlight ids with Neovim's API.
- buf_src_ids = {}
- for buf, definitions in defs_per_buf.items():
- buf_src_ids[buf] = []
- for definition in definitions:
- src_id = _add_highlight_definition(buf, definition)
- buf_src_ids[buf].append(src_id)
- for buf, src_ids in buf_src_ids.items():
- buf.vars['_jedi_usages_src_ids'] = src_ids
- else:
- for buf, definitions in defs_per_buf.items():
- try:
- for definition in definitions:
- _add_highlight_definition(buf, definition)
- except vim.error as exc:
- if exc.args[0].startswith('Vim:E275:'):
- # "Cannot add text property to unloaded buffer"
- _pending_definitions.setdefault(buf.name, []).extend(
- definitions)
- else:
- highlight_usages_for_vim_win()
- def _handle_pending_usages_for_buf():
- """Add (pending) highlights for the current buffer (Vim with textprops)."""
- buf = vim.current.buffer
- bufname = buf.name
- try:
- buf_defs = _pending_definitions[bufname]
- except KeyError:
- return
- for definition in buf_defs:
- _add_highlight_definition(buf, definition)
- del _pending_definitions[bufname]
- def _add_highlight_definition(buf, definition):
- lnum = definition.line
- start_col = definition.column
- # Skip highlighting of module definitions that point to the start
- # of the file.
- if definition.type == 'module' and lnum == 1 and start_col == 0:
- return
- _placed_definitions_in_buffers.add(buf)
- # TODO: validate that definition.name is at this position?
- # Would skip the module definitions from above already.
- length = len(definition.name)
- if vim_prop_add:
- # XXX: needs jediUsage highlight (via after/syntax/python.vim).
- global vim_prop_type_added
- if not vim_prop_type_added:
- vim.eval("prop_type_add('jediUsage', {'highlight': 'jediUsage'})")
- vim_prop_type_added = True
- vim_prop_add(lnum, start_col+1, {
- 'type': 'jediUsage',
- 'bufnr': buf.number,
- 'length': length,
- })
- return
- assert IS_NVIM
- end_col = definition.column + length
- src_id = buf.add_highlight('jediUsage', lnum-1, start_col, end_col,
- src_id=0)
- return src_id
- def highlight_usages_for_vim_win():
- """Highlight usages in the current window.
- It stores the matchids in a window-local variable.
- (matchaddpos() only works for the current window.)
- """
- global _current_definitions
- definitions = _current_definitions
- win = vim.current.window
- cur_matchids = win.vars.get('_jedi_usages_vim_matchids')
- if cur_matchids:
- if cur_matchids[0] == vim.current.buffer.number:
- return
- # Need to clear non-matching highlights.
- for matchid in cur_matchids[1:]:
- expr = 'matchdelete(%d)' % int(matchid)
- vim.eval(expr)
- matchids = []
- if definitions:
- buffer_path = vim.current.buffer.name
- for definition in definitions:
- if (definition.module_path or '') == buffer_path:
- positions = [
- [definition.line,
- definition.column + 1,
- len(definition.name)]
- ]
- expr = "matchaddpos('jediUsage', %s)" % repr(positions)
- matchids.append(int(vim_eval(expr)))
- if matchids:
- vim.current.window.vars['_jedi_usages_vim_matchids'] = [
- vim.current.buffer.number] + matchids
- elif cur_matchids is not None:
- # Always set it (uses an empty list for "unset", which is not possible
- # using del).
- vim.current.window.vars['_jedi_usages_vim_matchids'] = []
- # Remember if clearing is needed for later buffer autocommands.
- vim.current.buffer.vars['_jedi_usages_needs_clear'] = bool(matchids)
- @_check_jedi_availability(show_error=True)
- @catch_and_print_exceptions
- def show_documentation():
- script = get_script()
- try:
- definitions = script.goto_definitions()
- except Exception:
- # print to stdout, will be in :messages
- definitions = []
- print("Exception, this shouldn't happen.")
- print(traceback.format_exc())
- if not definitions:
- echo_highlight('No documentation found for that.')
- vim.command('return')
- return
- docs = []
- for d in definitions:
- doc = d.docstring()
- if doc:
- title = 'Docstring for %s' % d.desc_with_module
- underline = '=' * len(title)
- docs.append('%s\n%s\n%s' % (title, underline, doc))
- else:
- docs.append('|No Docstring for %s|' % d)
- text = ('\n' + '-' * 79 + '\n').join(docs)
- vim.command('let l:doc = %s' % repr(PythonToVimStr(text)))
- vim.command('let l:doc_lines = %s' % len(text.split('\n')))
- return True
- @catch_and_print_exceptions
- def clear_call_signatures():
- # Check if using command line call signatures
- if int(vim_eval("g:jedi#show_call_signatures")) == 2:
- vim_command('echo ""')
- return
- cursor = vim.current.window.cursor
- e = vim_eval('g:jedi#call_signature_escape')
- # We need two turns here to search and replace certain lines:
- # 1. Search for a line with a call signature and save the appended
- # characters
- # 2. Actually replace the line and redo the status quo.
- py_regex = r'%sjedi=([0-9]+), (.*?)%s.*?%sjedi%s'.replace(
- '%s', re.escape(e))
- for i, line in enumerate(vim.current.buffer):
- match = re.search(py_regex, line)
- if match is not None:
- # Some signs were added to minimize syntax changes due to call
- # signatures. We have to remove them again. The number of them is
- # specified in `match.group(1)`.
- after = line[match.end() + int(match.group(1)):]
- line = line[:match.start()] + match.group(2) + after
- vim.current.buffer[i] = line
- vim.current.window.cursor = cursor
- @_check_jedi_availability(show_error=False)
- @catch_and_print_exceptions
- def show_call_signatures(signatures=()):
- if int(vim_eval("has('conceal') && g:jedi#show_call_signatures")) == 0:
- return
- # We need to clear the signatures before we calculate them again. The
- # reason for this is that call signatures are unfortunately written to the
- # buffer.
- clear_call_signatures()
- if signatures == ():
- signatures = get_script().call_signatures()
- if not signatures:
- return
- if int(vim_eval("g:jedi#show_call_signatures")) == 2:
- return cmdline_call_signatures(signatures)
- seen_sigs = []
- for i, signature in enumerate(signatures):
- line, column = signature.bracket_start
- # signatures are listed above each other
- line_to_replace = line - i - 1
- # because there's a space before the bracket
- insert_column = column - 1
- if insert_column < 0 or line_to_replace <= 0:
- # Edge cases, when the call signature has no space on the screen.
- break
- # TODO check if completion menu is above or below
- line = vim_eval("getline(%s)" % line_to_replace)
- # Descriptions are usually looking like `param name`, remove the param.
- params = [p.description.replace('\n', '').replace('param ', '', 1)
- for p in signature.params]
- try:
- # *_*PLACEHOLDER*_* makes something fat. See after/syntax file.
- params[signature.index] = '*_*%s*_*' % params[signature.index]
- except (IndexError, TypeError):
- pass
- # Skip duplicates.
- if params in seen_sigs:
- continue
- seen_sigs.append(params)
- # This stuff is reaaaaally a hack! I cannot stress enough, that
- # this is a stupid solution. But there is really no other yet.
- # There is no possibility in VIM to draw on the screen, but there
- # will be one (see :help todo Patch to access screen under Python.
- # (Marko Mahni, 2010 Jul 18))
- text = " (%s) " % ', '.join(params)
- text = ' ' * (insert_column - len(line)) + text
- end_column = insert_column + len(text) - 2 # -2 due to bold symbols
- # Need to decode it with utf8, because vim returns always a python 2
- # string even if it is unicode.
- e = vim_eval('g:jedi#call_signature_escape')
- if hasattr(e, 'decode'):
- e = e.decode('UTF-8')
- # replace line before with cursor
- regex = "xjedi=%sx%sxjedix".replace('x', e)
- prefix, replace = line[:insert_column], line[insert_column:end_column]
- # Check the replace stuff for strings, to append them
- # (don't want to break the syntax)
- regex_quotes = r'''\\*["']+'''
- # `add` are all the quotation marks.
- # join them with a space to avoid producing '''
- add = ' '.join(re.findall(regex_quotes, replace))
- # search backwards
- if add and replace[0] in ['"', "'"]:
- a = re.search(regex_quotes + '$', prefix)
- add = ('' if a is None else a.group(0)) + add
- tup = '%s, %s' % (len(add), replace)
- repl = prefix + (regex % (tup, text)) + add + line[end_column:]
- vim_eval('setline(%s, %s)' % (line_to_replace, repr(PythonToVimStr(repl))))
- @catch_and_print_exceptions
- def cmdline_call_signatures(signatures):
- def get_params(s):
- return [p.description.replace('\n', '').replace('param ', '', 1) for p in s.params]
- def escape(string):
- return string.replace('"', '\\"').replace(r'\n', r'\\n')
- def join():
- return ', '.join(filter(None, (left, center, right)))
- def too_long():
- return len(join()) > max_msg_len
- if len(signatures) > 1:
- params = zip_longest(*map(get_params, signatures), fillvalue='_')
- params = ['(' + ', '.join(p) + ')' for p in params]
- else:
- params = get_params(signatures[0])
- index = next(iter(s.index for s in signatures if s.index is not None), None)
- # Allow 12 characters for showcmd plus 18 for ruler - setting
- # noruler/noshowcmd here causes incorrect undo history
- max_msg_len = int(vim_eval('&columns')) - 12
- if int(vim_eval('&ruler')):
- max_msg_len -= 18
- max_msg_len -= len(signatures[0].name) + 2 # call name + parentheses
- if max_msg_len < (1 if params else 0):
- return
- elif index is None:
- text = escape(', '.join(params))
- if params and len(text) > max_msg_len:
- text = ELLIPSIS
- elif max_msg_len < len(ELLIPSIS):
- return
- else:
- left = escape(', '.join(params[:index]))
- center = escape(params[index])
- right = escape(', '.join(params[index + 1:]))
- while too_long():
- if left and left != ELLIPSIS:
- left = ELLIPSIS
- continue
- if right and right != ELLIPSIS:
- right = ELLIPSIS
- continue
- if (left or right) and center != ELLIPSIS:
- left = right = None
- center = ELLIPSIS
- continue
- if too_long():
- # Should never reach here
- return
- max_num_spaces = max_msg_len
- if index is not None:
- max_num_spaces -= len(join())
- _, column = signatures[0].bracket_start
- spaces = min(int(vim_eval('g:jedi#first_col +'
- 'wincol() - col(".")')) +
- column - len(signatures[0].name),
- max_num_spaces) * ' '
- if index is not None:
- vim_command(' echon "%s" | '
- 'echohl Function | echon "%s" | '
- 'echohl None | echon "(" | '
- 'echohl jediFunction | echon "%s" | '
- 'echohl jediFat | echon "%s" | '
- 'echohl jediFunction | echon "%s" | '
- 'echohl None | echon ")"'
- % (spaces, signatures[0].name,
- left + ', ' if left else '',
- center, ', ' + right if right else ''))
- else:
- vim_command(' echon "%s" | '
- 'echohl Function | echon "%s" | '
- 'echohl None | echon "(%s)"'
- % (spaces, signatures[0].name, text))
- @_check_jedi_availability(show_error=True)
- @catch_and_print_exceptions
- def rename():
- if not int(vim.eval('a:0')):
- # Need to save the cursor position before insert mode
- cursor = vim.current.window.cursor
- changenr = vim.eval('changenr()') # track undo tree
- vim_command('augroup jedi_rename')
- vim_command('autocmd InsertLeave <buffer> call jedi#rename'
- '({}, {}, {})'.format(cursor[0], cursor[1], changenr))
- vim_command('augroup END')
- vim_command("let s:jedi_replace_orig = expand('<cword>')")
- line = vim_eval('getline(".")')
- vim_command('normal! diw')
- if re.match(r'\w+$', line[cursor[1]:]):
- # In case the deleted word is at the end of the line we need to
- # move the cursor to the end.
- vim_command('startinsert!')
- else:
- vim_command('startinsert')
- else:
- # Remove autocommand.
- vim_command('autocmd! jedi_rename InsertLeave')
- args = vim.eval('a:000')
- cursor = tuple(int(x) for x in args[:2])
- changenr = args[2]
- # Get replacement, if there is something on the cursor.
- # This won't be the case when the user ends insert mode right away,
- # and `<cword>` would pick up the nearest word instead.
- if vim_eval('getline(".")[getpos(".")[2]-1]') != ' ':
- replace = vim_eval("expand('<cword>')")
- else:
- replace = None
- vim_command('undo {}'.format(changenr))
- vim.current.window.cursor = cursor
- if replace:
- return do_rename(replace)
- def rename_visual():
- replace = vim.eval('input("Rename to: ")')
- orig = vim.eval('getline(".")[(getpos("\'<")[2]-1):getpos("\'>")[2]]')
- do_rename(replace, orig)
- def do_rename(replace, orig=None):
- if not len(replace):
- echo_highlight('No rename possible without name.')
- return
- if orig is None:
- orig = vim_eval('s:jedi_replace_orig')
- # Save original window / tab.
- saved_tab = int(vim_eval('tabpagenr()'))
- saved_win = int(vim_eval('winnr()'))
- temp_rename = usages(visuals=False)
- # Sort the whole thing reverse (positions at the end of the line
- # must be first, because they move the stuff before the position).
- temp_rename = sorted(temp_rename, reverse=True,
- key=lambda x: (x.module_path, x.line, x.column))
- buffers = set()
- for r in temp_rename:
- if r.in_builtin_module():
- continue
- if os.path.abspath(vim.current.buffer.name) != r.module_path:
- assert r.module_path is not None
- result = new_buffer(r.module_path)
- if not result:
- echo_highlight('Failed to create buffer window for %s!' % (
- r.module_path))
- continue
- buffers.add(vim.current.buffer.name)
- # Replace original word.
- r_line = vim.current.buffer[r.line - 1]
- vim.current.buffer[r.line - 1] = (r_line[:r.column] + replace +
- r_line[r.column + len(orig):])
- # Restore previous tab and window.
- vim_command('tabnext {0:d}'.format(saved_tab))
- vim_command('{0:d}wincmd w'.format(saved_win))
- if len(buffers) > 1:
- echo_highlight('Jedi did {0:d} renames in {1:d} buffers!'.format(
- len(temp_rename), len(buffers)))
- else:
- echo_highlight('Jedi did {0:d} renames!'.format(len(temp_rename)))
- @_check_jedi_availability(show_error=True)
- @catch_and_print_exceptions
- def py_import():
- # args are the same as for the :edit command
- args = shsplit(vim.eval('a:args'))
- import_path = args.pop()
- text = 'import %s' % import_path
- scr = jedi.Script(text, 1, len(text), '', environment=get_environment())
- try:
- completion = scr.goto_assignments()[0]
- except IndexError:
- echo_highlight('Cannot find %s in sys.path!' % import_path)
- else:
- if completion.column is None: # Python modules always have a line number.
- echo_highlight('%s is a builtin module.' % import_path)
- else:
- cmd_args = ' '.join([a.replace(' ', '\\ ') for a in args])
- new_buffer(completion.module_path, cmd_args)
- @catch_and_print_exceptions
- def py_import_completions():
- argl = vim.eval('a:argl')
- try:
- import jedi
- except ImportError:
- print('Pyimport completion requires jedi module: https://github.com/davidhalter/jedi')
- comps = []
- else:
- text = 'import %s' % argl
- script = jedi.Script(text, 1, len(text), '', environment=get_environment())
- comps = ['%s%s' % (argl, c.complete) for c in script.completions()]
- vim.command("return '%s'" % '\n'.join(comps))
- @catch_and_print_exceptions
- def new_buffer(path, options='', using_tagstack=False):
- # options are what you can to edit the edit options
- if int(vim_eval('g:jedi#use_tabs_not_buffers')) == 1:
- _tabnew(path, options)
- elif not vim_eval('g:jedi#use_splits_not_buffers') in [1, '1']:
- user_split_option = vim_eval('g:jedi#use_splits_not_buffers')
- split_options = {
- 'top': 'topleft split',
- 'left': 'topleft vsplit',
- 'right': 'botright vsplit',
- 'bottom': 'botright split',
- 'winwidth': 'vs'
- }
- if (user_split_option == 'winwidth' and
- vim.current.window.width <= 2 * int(vim_eval(
- "&textwidth ? &textwidth : 80"))):
- split_options['winwidth'] = 'sp'
- if user_split_option not in split_options:
- print('Unsupported value for g:jedi#use_splits_not_buffers: {0}. '
- 'Valid options are: {1}.'.format(
- user_split_option, ', '.join(split_options.keys())))
- else:
- vim_command(split_options[user_split_option] + " %s" % escape_file_path(path))
- else:
- if int(vim_eval("!&hidden && &modified")) == 1:
- if not vim_eval("bufname('%')"):
- echo_highlight('Cannot open a new buffer, use `:set hidden` or save your buffer')
- return False
- else:
- vim_command('w')
- if using_tagstack:
- return True
- vim_command('edit %s %s' % (options, escape_file_path(path)))
- # sometimes syntax is being disabled and the filetype not set.
- if int(vim_eval('!exists("g:syntax_on")')) == 1:
- vim_command('syntax enable')
- if int(vim_eval("&filetype != 'python'")) == 1:
- vim_command('set filetype=python')
- return True
- @catch_and_print_exceptions
- def _tabnew(path, options=''):
- """
- Open a file in a new tab or switch to an existing one.
- :param options: `:tabnew` options, read vim help.
- """
- path = os.path.abspath(path)
- if int(vim_eval('has("gui")')) == 1:
- vim_command('tab drop %s %s' % (options, escape_file_path(path)))
- return
- for tab_nr in range(int(vim_eval("tabpagenr('$')"))):
- for buf_nr in vim_eval("tabpagebuflist(%i + 1)" % tab_nr):
- buf_nr = int(buf_nr) - 1
- try:
- buf_path = vim.buffers[buf_nr].name
- except (LookupError, ValueError):
- # Just do good old asking for forgiveness.
- # don't know why this happens :-)
- pass
- else:
- if buf_path == path:
- # tab exists, just switch to that tab
- vim_command('tabfirst | tabnext %i' % (tab_nr + 1))
- # Goto the buffer's window.
- vim_command('exec bufwinnr(%i) . " wincmd w"' % (buf_nr + 1))
- break
- else:
- continue
- break
- else:
- # tab doesn't exist, add a new one.
- vim_command('tabnew %s' % escape_file_path(path))
- def escape_file_path(path):
- return path.replace(' ', r'\ ')
- def print_to_stdout(level, str_out):
- print(str_out)