2012-08-04 17:46:54 -07:00
|
|
|
#!/usr/bin/env python
|
|
|
|
#
|
2014-01-13 11:08:43 -08:00
|
|
|
# Copyright (C) 2011, 2012 Google Inc.
|
2012-08-04 17:46:54 -07:00
|
|
|
#
|
|
|
|
# This file is part of YouCompleteMe.
|
|
|
|
#
|
|
|
|
# YouCompleteMe 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.
|
|
|
|
#
|
|
|
|
# YouCompleteMe 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 YouCompleteMe. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
|
|
|
|
import vim
|
2013-10-03 10:14:31 -07:00
|
|
|
import os
|
2014-08-20 13:40:17 -07:00
|
|
|
import tempfile
|
2013-10-04 12:23:33 -07:00
|
|
|
import json
|
2015-08-28 08:26:18 -06:00
|
|
|
import re
|
2014-05-13 13:09:19 -07:00
|
|
|
from ycmd.utils import ToUtf8IfNeeded
|
|
|
|
from ycmd import user_options_store
|
2012-08-04 17:46:54 -07:00
|
|
|
|
2014-03-04 14:47:43 +04:00
|
|
|
BUFFER_COMMAND_MAP = { 'same-buffer' : 'edit',
|
|
|
|
'horizontal-split' : 'split',
|
|
|
|
'vertical-split' : 'vsplit',
|
|
|
|
'new-tab' : 'tabedit' }
|
|
|
|
|
2012-08-04 17:46:54 -07:00
|
|
|
def CurrentLineAndColumn():
|
2012-09-24 19:20:33 -07:00
|
|
|
"""Returns the 0-based current line and 0-based current column."""
|
2012-08-04 17:46:54 -07:00
|
|
|
# See the comment in CurrentColumn about the calculation for the line and
|
|
|
|
# column number
|
|
|
|
line, column = vim.current.window.cursor
|
|
|
|
line -= 1
|
|
|
|
return line, column
|
|
|
|
|
|
|
|
|
|
|
|
def CurrentColumn():
|
2012-08-15 19:39:03 -07:00
|
|
|
"""Returns the 0-based current column. Do NOT access the CurrentColumn in
|
2013-06-09 17:31:17 -07:00
|
|
|
vim.current.line. It doesn't exist yet when the cursor is at the end of the
|
|
|
|
line. Only the chars before the current column exist in vim.current.line."""
|
2012-08-04 17:46:54 -07:00
|
|
|
|
|
|
|
# vim's columns are 1-based while vim.current.line columns are 0-based
|
|
|
|
# ... but vim.current.window.cursor (which returns a (line, column) tuple)
|
|
|
|
# columns are 0-based, while the line from that same tuple is 1-based.
|
|
|
|
# vim.buffers buffer objects OTOH have 0-based lines and columns.
|
|
|
|
# Pigs have wings and I'm a loopy purple duck. Everything makes sense now.
|
|
|
|
return vim.current.window.cursor[ 1 ]
|
|
|
|
|
|
|
|
|
2014-08-28 11:36:46 -07:00
|
|
|
def CurrentLineContents():
|
|
|
|
return vim.current.line
|
|
|
|
|
|
|
|
|
2013-06-09 19:00:49 -07:00
|
|
|
def TextAfterCursor():
|
|
|
|
"""Returns the text after CurrentColumn."""
|
|
|
|
return vim.current.line[ CurrentColumn(): ]
|
|
|
|
|
|
|
|
|
2015-08-28 08:26:04 -06:00
|
|
|
def TextBeforeCursor():
|
|
|
|
"""Returns the text before CurrentColumn."""
|
|
|
|
return vim.current.line[ :CurrentColumn() ]
|
|
|
|
|
|
|
|
|
2014-05-28 15:48:49 -07:00
|
|
|
# Expects version_string in 'MAJOR.MINOR.PATCH' format, e.g. '7.4.301'
|
|
|
|
def VimVersionAtLeast( version_string ):
|
|
|
|
major, minor, patch = [ int( x ) for x in version_string.split( '.' ) ]
|
|
|
|
|
|
|
|
# For Vim 7.4.301, v:version is '704'
|
|
|
|
actual_major_and_minor = GetIntValue( 'v:version' )
|
2015-08-28 08:26:04 -06:00
|
|
|
matching_major_and_minor = major * 100 + minor
|
|
|
|
if actual_major_and_minor != matching_major_and_minor:
|
|
|
|
return actual_major_and_minor > matching_major_and_minor
|
2014-05-28 15:48:49 -07:00
|
|
|
|
|
|
|
return GetBoolValue( 'has("patch{0}")'.format( patch ) )
|
|
|
|
|
|
|
|
|
2013-09-05 23:43:14 -07:00
|
|
|
# Note the difference between buffer OPTIONS and VARIABLES; the two are not
|
|
|
|
# the same.
|
|
|
|
def GetBufferOption( buffer_object, option ):
|
2014-01-06 15:00:51 -08:00
|
|
|
# NOTE: We used to check for the 'options' property on the buffer_object which
|
|
|
|
# is available in recent versions of Vim and would then use:
|
|
|
|
#
|
|
|
|
# buffer_object.options[ option ]
|
|
|
|
#
|
|
|
|
# to read the value, BUT this caused annoying flickering when the
|
|
|
|
# buffer_object was a hidden buffer (with option = 'ft'). This was all due to
|
|
|
|
# a Vim bug. Until this is fixed, we won't use it.
|
2012-08-04 17:46:54 -07:00
|
|
|
|
2013-09-16 13:55:41 -07:00
|
|
|
to_eval = 'getbufvar({0}, "&{1}")'.format( buffer_object.number, option )
|
2013-09-05 23:43:14 -07:00
|
|
|
return GetVariableValue( to_eval )
|
|
|
|
|
|
|
|
|
2014-02-23 17:50:51 +04:00
|
|
|
def BufferModified( buffer_object ):
|
2014-03-23 12:33:27 +04:00
|
|
|
return bool( int( GetBufferOption( buffer_object, 'mod' ) ) )
|
2014-03-22 14:24:16 +04:00
|
|
|
|
2013-09-05 23:43:14 -07:00
|
|
|
|
2014-02-23 17:50:51 +04:00
|
|
|
def GetUnsavedAndCurrentBufferData():
|
2013-09-05 23:43:14 -07:00
|
|
|
buffers_data = {}
|
|
|
|
for buffer_object in vim.buffers:
|
|
|
|
if not ( BufferModified( buffer_object ) or
|
|
|
|
buffer_object == vim.current.buffer ):
|
|
|
|
continue
|
|
|
|
|
2013-10-06 19:45:47 -07:00
|
|
|
buffers_data[ GetBufferFilepath( buffer_object ) ] = {
|
2015-04-14 10:23:35 -07:00
|
|
|
# Add a newline to match what gets saved to disk. See #1455 for details.
|
|
|
|
'contents': '\n'.join( buffer_object ) + '\n',
|
2013-09-05 23:43:14 -07:00
|
|
|
'filetypes': FiletypesForBuffer( buffer_object )
|
|
|
|
}
|
|
|
|
|
|
|
|
return buffers_data
|
2012-08-04 17:46:54 -07:00
|
|
|
|
|
|
|
|
2013-10-03 10:14:31 -07:00
|
|
|
def GetBufferNumberForFilename( filename, open_file_if_needed = True ):
|
2014-07-07 22:52:17 +07:00
|
|
|
return GetIntValue( u"bufnr('{0}', {1})".format(
|
2014-07-31 10:28:44 -07:00
|
|
|
EscapeForVim( os.path.realpath( filename ) ),
|
2014-01-04 14:28:27 -08:00
|
|
|
int( open_file_if_needed ) ) )
|
2013-10-03 10:14:31 -07:00
|
|
|
|
|
|
|
|
2013-10-06 19:45:47 -07:00
|
|
|
def GetCurrentBufferFilepath():
|
|
|
|
return GetBufferFilepath( vim.current.buffer )
|
|
|
|
|
|
|
|
|
2014-01-04 14:28:27 -08:00
|
|
|
def BufferIsVisible( buffer_number ):
|
|
|
|
if buffer_number < 0:
|
|
|
|
return False
|
|
|
|
window_number = GetIntValue( "bufwinnr({0})".format( buffer_number ) )
|
|
|
|
return window_number != -1
|
|
|
|
|
|
|
|
|
2013-10-06 19:45:47 -07:00
|
|
|
def GetBufferFilepath( buffer_object ):
|
|
|
|
if buffer_object.name:
|
|
|
|
return buffer_object.name
|
|
|
|
# Buffers that have just been created by a command like :enew don't have any
|
2014-08-20 13:40:17 -07:00
|
|
|
# buffer name so we use the buffer number for that. Also, os.getcwd() throws
|
|
|
|
# an exception when the CWD has been deleted so we handle that.
|
|
|
|
try:
|
|
|
|
folder_path = os.getcwd()
|
|
|
|
except OSError:
|
|
|
|
folder_path = tempfile.gettempdir()
|
|
|
|
return os.path.join( folder_path, str( buffer_object.number ) )
|
2013-10-06 19:45:47 -07:00
|
|
|
|
|
|
|
|
2014-09-14 01:15:40 +07:00
|
|
|
def UnplaceSignInBuffer( buffer_number, sign_id ):
|
2014-01-04 14:28:27 -08:00
|
|
|
if buffer_number < 0:
|
|
|
|
return
|
2014-09-14 01:15:40 +07:00
|
|
|
vim.command(
|
|
|
|
'try | exec "sign unplace {0} buffer={1}" | catch /E158/ | endtry'.format(
|
|
|
|
sign_id, buffer_number ) )
|
2014-01-04 14:28:27 -08:00
|
|
|
|
|
|
|
|
|
|
|
def PlaceSign( sign_id, line_num, buffer_num, is_error = True ):
|
2014-06-09 10:18:03 -07:00
|
|
|
# libclang can give us diagnostics that point "outside" the file; Vim borks
|
|
|
|
# on these.
|
|
|
|
if line_num < 1:
|
|
|
|
line_num = 1
|
|
|
|
|
2014-01-04 14:28:27 -08:00
|
|
|
sign_name = 'YcmError' if is_error else 'YcmWarning'
|
|
|
|
vim.command( 'sign place {0} line={1} name={2} buffer={3}'.format(
|
|
|
|
sign_id, line_num, sign_name, buffer_num ) )
|
|
|
|
|
|
|
|
|
2014-09-14 01:15:40 +07:00
|
|
|
def PlaceDummySign( sign_id, buffer_num, line_num ):
|
|
|
|
if buffer_num < 0 or line_num < 0:
|
|
|
|
return
|
|
|
|
vim.command( 'sign define ycm_dummy_sign' )
|
|
|
|
vim.command(
|
|
|
|
'sign place {0} name=ycm_dummy_sign line={1} buffer={2}'.format(
|
|
|
|
sign_id,
|
|
|
|
line_num,
|
|
|
|
buffer_num,
|
|
|
|
)
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
def UnPlaceDummySign( sign_id, buffer_num ):
|
|
|
|
if buffer_num < 0:
|
|
|
|
return
|
|
|
|
vim.command( 'sign undefine ycm_dummy_sign' )
|
|
|
|
vim.command( 'sign unplace {0} buffer={1}'.format( sign_id, buffer_num ) )
|
|
|
|
|
|
|
|
|
2014-01-04 18:32:42 -08:00
|
|
|
def ClearYcmSyntaxMatches():
|
|
|
|
matches = VimExpressionToPythonType( 'getmatches()' )
|
|
|
|
for match in matches:
|
|
|
|
if match[ 'group' ].startswith( 'Ycm' ):
|
|
|
|
vim.eval( 'matchdelete({0})'.format( match[ 'id' ] ) )
|
|
|
|
|
|
|
|
|
|
|
|
# Returns the ID of the newly added match
|
2014-01-09 15:48:48 -08:00
|
|
|
# Both line and column numbers are 1-based
|
2014-01-05 13:30:22 -08:00
|
|
|
def AddDiagnosticSyntaxMatch( line_num,
|
|
|
|
column_num,
|
|
|
|
line_end_num = None,
|
|
|
|
column_end_num = None,
|
|
|
|
is_error = True ):
|
2014-01-04 18:32:42 -08:00
|
|
|
group = 'YcmErrorSection' if is_error else 'YcmWarningSection'
|
2014-01-05 13:30:22 -08:00
|
|
|
|
|
|
|
if not line_end_num:
|
|
|
|
line_end_num = line_num
|
|
|
|
|
2014-01-09 15:48:48 -08:00
|
|
|
line_num, column_num = LineAndColumnNumbersClamped( line_num, column_num )
|
|
|
|
line_end_num, column_end_num = LineAndColumnNumbersClamped( line_end_num,
|
|
|
|
column_end_num )
|
|
|
|
|
2014-01-05 13:30:22 -08:00
|
|
|
if not column_end_num:
|
|
|
|
return GetIntValue(
|
|
|
|
"matchadd('{0}', '\%{1}l\%{2}c')".format( group, line_num, column_num ) )
|
|
|
|
else:
|
|
|
|
return GetIntValue(
|
2014-05-19 11:10:03 -07:00
|
|
|
"matchadd('{0}', '\%{1}l\%{2}c\_.\\{{-}}\%{3}l\%{4}c')".format(
|
2014-01-05 13:30:22 -08:00
|
|
|
group, line_num, column_num, line_end_num, column_end_num ) )
|
2014-01-04 18:32:42 -08:00
|
|
|
|
|
|
|
|
2014-01-09 15:48:48 -08:00
|
|
|
# Clamps the line and column numbers so that they are not past the contents of
|
|
|
|
# the buffer. Numbers are 1-based.
|
|
|
|
def LineAndColumnNumbersClamped( line_num, column_num ):
|
|
|
|
new_line_num = line_num
|
|
|
|
new_column_num = column_num
|
|
|
|
|
|
|
|
max_line = len( vim.current.buffer )
|
|
|
|
if line_num and line_num > max_line:
|
|
|
|
new_line_num = max_line
|
|
|
|
|
|
|
|
max_column = len( vim.current.buffer[ new_line_num - 1 ] )
|
|
|
|
if column_num and column_num > max_column:
|
|
|
|
new_column_num = max_column
|
|
|
|
|
|
|
|
return new_line_num, new_column_num
|
|
|
|
|
|
|
|
|
2014-01-08 19:09:40 -08:00
|
|
|
def SetLocationList( diagnostics ):
|
|
|
|
"""Diagnostics should be in qflist format; see ":h setqflist" for details."""
|
|
|
|
vim.eval( 'setloclist( 0, {0} )'.format( json.dumps( diagnostics ) ) )
|
|
|
|
|
|
|
|
|
|
|
|
def ConvertDiagnosticsToQfList( diagnostics ):
|
|
|
|
def ConvertDiagnosticToQfFormat( diagnostic ):
|
|
|
|
# see :h getqflist for a description of the dictionary fields
|
|
|
|
# Note that, as usual, Vim is completely inconsistent about whether
|
|
|
|
# line/column numbers are 1 or 0 based in its various APIs. Here, it wants
|
|
|
|
# them to be 1-based.
|
|
|
|
location = diagnostic[ 'location' ]
|
2014-06-09 10:03:40 -07:00
|
|
|
line_num = location[ 'line_num' ]
|
2014-05-29 13:25:43 -07:00
|
|
|
|
|
|
|
# libclang can give us diagnostics that point "outside" the file; Vim borks
|
|
|
|
# on these.
|
|
|
|
if line_num < 1:
|
|
|
|
line_num = 1
|
|
|
|
|
2015-08-05 22:09:07 +01:00
|
|
|
text = diagnostic[ 'text' ]
|
|
|
|
if diagnostic.get( 'fixit_available', False ):
|
|
|
|
text += ' (FixIt available)'
|
|
|
|
|
2014-01-08 19:09:40 -08:00
|
|
|
return {
|
|
|
|
'bufnr' : GetBufferNumberForFilename( location[ 'filepath' ] ),
|
2014-05-29 13:25:43 -07:00
|
|
|
'lnum' : line_num,
|
2014-06-09 10:03:40 -07:00
|
|
|
'col' : location[ 'column_num' ],
|
2015-08-05 22:09:07 +01:00
|
|
|
'text' : ToUtf8IfNeeded( text ),
|
2014-06-03 14:49:53 -07:00
|
|
|
'type' : diagnostic[ 'kind' ][ 0 ],
|
2014-01-08 19:09:40 -08:00
|
|
|
'valid' : 1
|
|
|
|
}
|
|
|
|
|
|
|
|
return [ ConvertDiagnosticToQfFormat( x ) for x in diagnostics ]
|
|
|
|
|
|
|
|
|
2013-10-04 12:23:33 -07:00
|
|
|
# Given a dict like {'a': 1}, loads it into Vim as if you ran 'let g:a = 1'
|
|
|
|
# When |overwrite| is True, overwrites the existing value in Vim.
|
|
|
|
def LoadDictIntoVimGlobals( new_globals, overwrite = True ):
|
|
|
|
extend_option = '"force"' if overwrite else '"keep"'
|
|
|
|
|
|
|
|
# We need to use json.dumps because that won't use the 'u' prefix on strings
|
|
|
|
# which Vim would bork on.
|
2013-10-04 15:44:16 -07:00
|
|
|
vim.eval( 'extend( g:, {0}, {1})'.format( json.dumps( new_globals ),
|
|
|
|
extend_option ) )
|
2013-10-04 12:23:33 -07:00
|
|
|
|
|
|
|
|
|
|
|
# Changing the returned dict will NOT change the value in Vim.
|
2013-10-06 18:26:59 -07:00
|
|
|
def GetReadOnlyVimGlobals( force_python_objects = False ):
|
|
|
|
if force_python_objects:
|
|
|
|
return vim.eval( 'g:' )
|
|
|
|
|
2013-10-04 12:23:33 -07:00
|
|
|
try:
|
|
|
|
# vim.vars is fairly new so it might not exist
|
|
|
|
return vim.vars
|
|
|
|
except:
|
|
|
|
return vim.eval( 'g:' )
|
|
|
|
|
|
|
|
|
2013-10-26 16:22:43 -07:00
|
|
|
def VimExpressionToPythonType( vim_expression ):
|
|
|
|
result = vim.eval( vim_expression )
|
|
|
|
if not isinstance( result, basestring ):
|
|
|
|
return result
|
|
|
|
try:
|
|
|
|
return int( result )
|
|
|
|
except ValueError:
|
|
|
|
return result
|
|
|
|
|
|
|
|
|
2014-03-22 14:24:16 +04:00
|
|
|
def HiddenEnabled( buffer_object ):
|
2014-03-23 12:33:27 +04:00
|
|
|
return bool( int( GetBufferOption( buffer_object, 'hid' ) ) )
|
2014-03-22 14:24:16 +04:00
|
|
|
|
|
|
|
|
|
|
|
def BufferIsUsable( buffer_object ):
|
|
|
|
return not BufferModified( buffer_object ) or HiddenEnabled( buffer_object )
|
|
|
|
|
|
|
|
|
2014-05-19 12:37:30 -07:00
|
|
|
def EscapedFilepath( filepath ):
|
|
|
|
return filepath.replace( ' ' , r'\ ' )
|
|
|
|
|
|
|
|
|
2015-03-23 02:07:35 +01:00
|
|
|
# Both |line| and |column| need to be 1-based
|
|
|
|
def TryJumpLocationInOpenedTab( filename, line, column ):
|
|
|
|
filepath = os.path.realpath( filename )
|
|
|
|
|
|
|
|
for tab in vim.tabpages:
|
|
|
|
for win in tab.windows:
|
|
|
|
if win.buffer.name == filepath:
|
|
|
|
vim.current.tabpage = tab
|
|
|
|
vim.current.window = win
|
|
|
|
vim.current.window.cursor = ( line, column - 1 )
|
|
|
|
|
|
|
|
# Center the screen on the jumped-to location
|
|
|
|
vim.command( 'normal! zz' )
|
2015-10-14 11:37:33 +04:00
|
|
|
return True
|
|
|
|
# 'filename' is not opened in any tab pages
|
|
|
|
return False
|
|
|
|
|
|
|
|
|
2015-10-27 17:00:11 +01:00
|
|
|
# Maps User command to vim command
|
|
|
|
def GetVimCommand( user_command, default = 'edit' ):
|
|
|
|
vim_command = BUFFER_COMMAND_MAP.get( user_command, default )
|
2015-10-14 11:37:33 +04:00
|
|
|
if vim_command == 'edit' and not BufferIsUsable( vim.current.buffer ):
|
|
|
|
vim_command = 'split'
|
|
|
|
return vim_command
|
2015-03-23 02:07:35 +01:00
|
|
|
|
|
|
|
|
2013-03-31 20:38:29 -07:00
|
|
|
# Both |line| and |column| need to be 1-based
|
|
|
|
def JumpToLocation( filename, line, column ):
|
|
|
|
# Add an entry to the jumplist
|
2013-03-31 21:01:38 -07:00
|
|
|
vim.command( "normal! m'" )
|
2013-03-31 20:38:29 -07:00
|
|
|
|
2013-10-06 19:45:47 -07:00
|
|
|
if filename != GetCurrentBufferFilepath():
|
2013-03-31 20:38:29 -07:00
|
|
|
# We prefix the command with 'keepjumps' so that opening the file is not
|
|
|
|
# recorded in the jumplist. So when we open the file and move the cursor to
|
|
|
|
# a location in it, the user can use CTRL-O to jump back to the original
|
|
|
|
# location, not to the start of the newly opened file.
|
|
|
|
# Sadly this fails on random occasions and the undesired jump remains in the
|
|
|
|
# jumplist.
|
2014-03-04 14:47:43 +04:00
|
|
|
user_command = user_options_store.Value( 'goto_buffer_command' )
|
2015-03-23 02:07:35 +01:00
|
|
|
|
|
|
|
if user_command == 'new-or-existing-tab':
|
2015-10-14 11:37:33 +04:00
|
|
|
if TryJumpLocationInOpenedTab( filename, line, column ):
|
2015-03-23 02:07:35 +01:00
|
|
|
return
|
2015-10-14 11:37:33 +04:00
|
|
|
user_command = 'new-tab'
|
2015-03-23 02:07:35 +01:00
|
|
|
|
2015-10-27 17:00:11 +01:00
|
|
|
vim_command = GetVimCommand( user_command )
|
2015-10-13 18:46:20 +04:00
|
|
|
try:
|
2015-10-14 11:37:33 +04:00
|
|
|
vim.command( 'keepjumps {0} {1}'.format( vim_command,
|
2015-10-13 18:46:20 +04:00
|
|
|
EscapedFilepath( filename ) ) )
|
|
|
|
# When the file we are trying to jump to has a swap file
|
|
|
|
# Vim opens swap-exists-choices dialog and throws vim.error with E325 error,
|
|
|
|
# or KeyboardInterrupt after user selects one of the options.
|
|
|
|
except vim.error as e:
|
2015-10-14 11:37:33 +04:00
|
|
|
if 'E325' not in str( e ):
|
2015-10-13 18:46:20 +04:00
|
|
|
raise
|
2015-10-14 11:37:33 +04:00
|
|
|
# Do nothing if the target file is still not opened (user chose (Q)uit)
|
|
|
|
if filename != GetCurrentBufferFilepath():
|
|
|
|
return
|
2015-10-13 18:46:20 +04:00
|
|
|
# Thrown when user chooses (A)bort in .swp message box
|
|
|
|
except KeyboardInterrupt:
|
|
|
|
return
|
2013-03-31 20:38:29 -07:00
|
|
|
vim.current.window.cursor = ( line, column - 1 )
|
|
|
|
|
|
|
|
# Center the screen on the jumped-to location
|
|
|
|
vim.command( 'normal! zz' )
|
|
|
|
|
|
|
|
|
2013-09-05 23:43:14 -07:00
|
|
|
def NumLinesInBuffer( buffer_object ):
|
2012-08-04 17:46:54 -07:00
|
|
|
# This is actually less than obvious, that's why it's wrapped in a function
|
2013-09-05 23:43:14 -07:00
|
|
|
return len( buffer_object )
|
2012-08-04 17:46:54 -07:00
|
|
|
|
|
|
|
|
2015-06-20 04:46:26 +02:00
|
|
|
# Calling this function from the non-GUI thread will sometimes crash Vim. At
|
|
|
|
# the time of writing, YCM only uses the GUI thread inside Vim (this used to
|
|
|
|
# not be the case).
|
|
|
|
# We redraw the screen before displaying the message to avoid the "Press ENTER
|
|
|
|
# or type command to continue" prompt when editing a new C-family file.
|
2012-08-04 17:46:54 -07:00
|
|
|
def PostVimMessage( message ):
|
2015-06-20 04:46:26 +02:00
|
|
|
vim.command( "redraw | echohl WarningMsg | echom '{0}' | echohl None"
|
2013-10-15 11:19:56 -07:00
|
|
|
.format( EscapeForVim( str( message ) ) ) )
|
|
|
|
|
2014-01-04 14:28:27 -08:00
|
|
|
|
2013-10-15 11:19:56 -07:00
|
|
|
# Unlike PostVimMesasge, this supports messages with newlines in them because it
|
|
|
|
# uses 'echo' instead of 'echomsg'. This also means that the message will NOT
|
|
|
|
# appear in Vim's message log.
|
|
|
|
def PostMultiLineNotice( message ):
|
|
|
|
vim.command( "echohl WarningMsg | echo '{0}' | echohl None"
|
2013-09-27 13:52:04 -07:00
|
|
|
.format( EscapeForVim( str( message ) ) ) )
|
2012-08-04 17:46:54 -07:00
|
|
|
|
|
|
|
|
2013-02-25 10:49:51 +01:00
|
|
|
def PresentDialog( message, choices, default_choice_index = 0 ):
|
|
|
|
"""Presents the user with a dialog where a choice can be made.
|
|
|
|
This will be a dialog for gvim users or a question in the message buffer
|
|
|
|
for vim users or if `set guioptions+=c` was used.
|
|
|
|
|
|
|
|
choices is list of alternatives.
|
|
|
|
default_choice_index is the 0-based index of the default element
|
|
|
|
that will get choosen if the user hits <CR>. Use -1 for no default.
|
|
|
|
|
|
|
|
PresentDialog will return a 0-based index into the list
|
|
|
|
or -1 if the dialog was dismissed by using <Esc>, Ctrl-C, etc.
|
|
|
|
|
|
|
|
See also:
|
|
|
|
:help confirm() in vim (Note that vim uses 1-based indexes)
|
|
|
|
|
|
|
|
Example call:
|
|
|
|
PresentDialog("Is this a nice example?", ["Yes", "No", "May&be"])
|
|
|
|
Is this a nice example?
|
|
|
|
[Y]es, (N)o, May(b)e:"""
|
|
|
|
to_eval = "confirm('{0}', '{1}', {2})".format( EscapeForVim( message ),
|
|
|
|
EscapeForVim( "\n" .join( choices ) ), default_choice_index + 1 )
|
|
|
|
return int( vim.eval( to_eval ) ) - 1
|
|
|
|
|
|
|
|
|
|
|
|
def Confirm( message ):
|
|
|
|
return bool( PresentDialog( message, [ "Ok", "Cancel" ] ) == 0 )
|
|
|
|
|
|
|
|
|
2014-01-04 16:45:34 -08:00
|
|
|
def EchoText( text, log_as_message = True ):
|
2013-02-21 10:50:38 -08:00
|
|
|
def EchoLine( text ):
|
2014-01-04 16:45:34 -08:00
|
|
|
command = 'echom' if log_as_message else 'echo'
|
|
|
|
vim.command( "{0} '{1}'".format( command, EscapeForVim( text ) ) )
|
2013-02-21 10:50:38 -08:00
|
|
|
|
2014-01-05 16:09:40 -08:00
|
|
|
for line in str( text ).split( '\n' ):
|
2013-02-21 10:50:38 -08:00
|
|
|
EchoLine( line )
|
2012-08-15 19:39:03 -07:00
|
|
|
|
|
|
|
|
2014-01-09 14:11:15 -08:00
|
|
|
# Echos text but truncates it so that it all fits on one line
|
2014-01-04 16:45:34 -08:00
|
|
|
def EchoTextVimWidth( text ):
|
|
|
|
vim_width = GetIntValue( '&columns' )
|
2014-01-13 10:00:05 -08:00
|
|
|
truncated_text = ToUtf8IfNeeded( text )[ : int( vim_width * 0.9 ) ]
|
2014-01-04 16:45:34 -08:00
|
|
|
truncated_text.replace( '\n', ' ' )
|
2014-01-09 13:49:06 -08:00
|
|
|
|
|
|
|
old_ruler = GetIntValue( '&ruler' )
|
|
|
|
old_showcmd = GetIntValue( '&showcmd' )
|
|
|
|
vim.command( 'set noruler noshowcmd' )
|
|
|
|
|
2014-01-04 16:45:34 -08:00
|
|
|
EchoText( truncated_text, False )
|
|
|
|
|
2014-01-09 13:49:06 -08:00
|
|
|
vim.command( 'let &ruler = {0}'.format( old_ruler ) )
|
|
|
|
vim.command( 'let &showcmd = {0}'.format( old_showcmd ) )
|
|
|
|
|
2014-01-04 16:45:34 -08:00
|
|
|
|
2012-08-04 17:46:54 -07:00
|
|
|
def EscapeForVim( text ):
|
|
|
|
return text.replace( "'", "''" )
|
|
|
|
|
|
|
|
|
2013-01-31 16:19:56 -08:00
|
|
|
def CurrentFiletypes():
|
2013-09-05 23:43:14 -07:00
|
|
|
return vim.eval( "&filetype" ).split( '.' )
|
2012-08-05 13:12:10 -07:00
|
|
|
|
|
|
|
|
2013-03-03 10:48:34 -08:00
|
|
|
def FiletypesForBuffer( buffer_object ):
|
|
|
|
# NOTE: Getting &ft for other buffers only works when the buffer has been
|
|
|
|
# visited by the user at least once, which is true for modified buffers
|
2013-09-05 23:43:14 -07:00
|
|
|
return GetBufferOption( buffer_object, 'ft' ).split( '.' )
|
2013-03-03 10:48:34 -08:00
|
|
|
|
|
|
|
|
2015-08-28 08:26:18 -06:00
|
|
|
def VariableExists( variable ):
|
|
|
|
return GetBoolValue( "exists( '{0}' )".format( EscapeForVim( variable ) ) )
|
|
|
|
|
|
|
|
|
|
|
|
def SetVariableValue( variable, value ):
|
|
|
|
vim.command( "let {0} = '{1}'".format( variable, EscapeForVim( value ) ) )
|
|
|
|
|
|
|
|
|
2012-08-05 13:12:10 -07:00
|
|
|
def GetVariableValue( variable ):
|
|
|
|
return vim.eval( variable )
|
2013-02-25 10:49:17 +01:00
|
|
|
|
|
|
|
|
|
|
|
def GetBoolValue( variable ):
|
|
|
|
return bool( int( vim.eval( variable ) ) )
|
2013-09-02 14:45:53 -07:00
|
|
|
|
|
|
|
|
|
|
|
def GetIntValue( variable ):
|
|
|
|
return int( vim.eval( variable ) )
|
2013-10-22 10:51:37 -07:00
|
|
|
|
2015-08-05 22:09:07 +01:00
|
|
|
|
2015-09-12 16:30:21 +02:00
|
|
|
def ReplaceChunksList( chunks, vim_buffer = None ):
|
|
|
|
if vim_buffer is None:
|
|
|
|
vim_buffer = vim.current.buffer
|
|
|
|
|
|
|
|
# We need to track the difference in length, but ensuring we apply fixes
|
|
|
|
# in ascending order of insertion point.
|
|
|
|
chunks.sort( key = lambda chunk: (
|
2015-09-12 17:08:58 +02:00
|
|
|
chunk[ 'range' ][ 'start' ][ 'line_num' ],
|
|
|
|
chunk[ 'range' ][ 'start' ][ 'column_num' ]
|
2015-09-12 16:30:21 +02:00
|
|
|
) )
|
|
|
|
|
|
|
|
# Remember the line number we're processing. Negative line number means we
|
|
|
|
# haven't processed any lines yet (by nature of being not equal to any
|
|
|
|
# real line number).
|
|
|
|
last_line = -1
|
|
|
|
|
|
|
|
line_delta = 0
|
|
|
|
for chunk in chunks:
|
|
|
|
if chunk[ 'range' ][ 'start' ][ 'line_num' ] != last_line:
|
|
|
|
# If this chunk is on a different line than the previous chunk,
|
|
|
|
# then ignore previous deltas (as offsets won't have changed).
|
|
|
|
last_line = chunk[ 'range' ][ 'end' ][ 'line_num' ]
|
|
|
|
char_delta = 0
|
|
|
|
|
|
|
|
( new_line_delta, new_char_delta ) = ReplaceChunk(
|
|
|
|
chunk[ 'range' ][ 'start' ],
|
|
|
|
chunk[ 'range' ][ 'end' ],
|
|
|
|
chunk[ 'replacement_text' ],
|
|
|
|
line_delta, char_delta,
|
|
|
|
vim_buffer )
|
|
|
|
line_delta += new_line_delta
|
|
|
|
char_delta += new_char_delta
|
|
|
|
|
|
|
|
|
2015-08-05 22:09:07 +01:00
|
|
|
# Replace the chunk of text specified by a contiguous range with the supplied
|
|
|
|
# text.
|
|
|
|
# * start and end are objects with line_num and column_num properties
|
|
|
|
# * the range is inclusive
|
|
|
|
# * indices are all 1-based
|
|
|
|
# * the returned character delta is the delta for the last line
|
|
|
|
#
|
|
|
|
# returns the delta (in lines and characters) that any position after the end
|
|
|
|
# needs to be adjusted by.
|
|
|
|
def ReplaceChunk( start, end, replacement_text, line_delta, char_delta,
|
2015-09-12 16:30:21 +02:00
|
|
|
vim_buffer ):
|
2015-08-05 22:09:07 +01:00
|
|
|
# ycmd's results are all 1-based, but vim's/python's are all 0-based
|
|
|
|
# (so we do -1 on all of the values)
|
|
|
|
start_line = start[ 'line_num' ] - 1 + line_delta
|
|
|
|
end_line = end[ 'line_num' ] - 1 + line_delta
|
|
|
|
source_lines_count = end_line - start_line + 1
|
|
|
|
start_column = start[ 'column_num' ] - 1 + char_delta
|
|
|
|
end_column = end[ 'column_num' ] - 1
|
|
|
|
if source_lines_count == 1:
|
|
|
|
end_column += char_delta
|
|
|
|
|
|
|
|
replacement_lines = replacement_text.splitlines( False )
|
|
|
|
if not replacement_lines:
|
|
|
|
replacement_lines = [ '' ]
|
|
|
|
replacement_lines_count = len( replacement_lines )
|
|
|
|
|
|
|
|
end_existing_text = vim_buffer[ end_line ][ end_column : ]
|
|
|
|
start_existing_text = vim_buffer[ start_line ][ : start_column ]
|
|
|
|
|
|
|
|
new_char_delta = ( len( replacement_lines[ -1 ] )
|
|
|
|
- ( end_column - start_column ) )
|
|
|
|
if replacement_lines_count > 1:
|
|
|
|
new_char_delta -= start_column
|
|
|
|
|
|
|
|
replacement_lines[ 0 ] = start_existing_text + replacement_lines[ 0 ]
|
|
|
|
replacement_lines[ -1 ] = replacement_lines[ -1 ] + end_existing_text
|
|
|
|
|
|
|
|
vim_buffer[ start_line : end_line + 1 ] = replacement_lines[:]
|
|
|
|
|
|
|
|
new_line_delta = replacement_lines_count - source_lines_count
|
|
|
|
return ( new_line_delta, new_char_delta )
|
2015-08-28 08:26:18 -06:00
|
|
|
|
|
|
|
|
|
|
|
def InsertNamespace( namespace ):
|
2015-08-31 10:51:23 -06:00
|
|
|
if VariableExists( 'g:ycm_csharp_insert_namespace_expr' ):
|
|
|
|
expr = GetVariableValue( 'g:ycm_csharp_insert_namespace_expr' )
|
|
|
|
if expr:
|
|
|
|
SetVariableValue( "g:ycm_namespace_to_insert", namespace )
|
|
|
|
vim.eval( expr )
|
|
|
|
return
|
|
|
|
|
|
|
|
pattern = '^\s*using\(\s\+[a-zA-Z0-9]\+\s\+=\)\?\s\+[a-zA-Z0-9.]\+\s*;\s*'
|
|
|
|
line = SearchInCurrentBuffer( pattern )
|
|
|
|
existing_line = LineTextInCurrentBuffer( line )
|
|
|
|
existing_indent = re.sub( r"\S.*", "", existing_line )
|
|
|
|
new_line = "{0}using {1};\n\n".format( existing_indent, namespace )
|
|
|
|
replace_pos = { 'line_num': line + 1, 'column_num': 1 }
|
|
|
|
ReplaceChunk( replace_pos, replace_pos, new_line, 0, 0 )
|
|
|
|
PostVimMessage( "Add namespace: {0}".format( namespace ) )
|
2015-08-28 08:26:18 -06:00
|
|
|
|
|
|
|
|
|
|
|
def SearchInCurrentBuffer( pattern ):
|
|
|
|
return GetIntValue( "search('{0}', 'Wcnb')".format( EscapeForVim( pattern )))
|
|
|
|
|
|
|
|
|
|
|
|
def LineTextInCurrentBuffer( line ):
|
|
|
|
return vim.current.buffer[ line ]
|
2015-09-06 20:07:42 +01:00
|
|
|
|
|
|
|
|
|
|
|
def ClosePreviewWindow():
|
|
|
|
""" Close the preview window if it is present, otherwise do nothing """
|
|
|
|
vim.command( 'silent! pclose!' )
|
|
|
|
|
|
|
|
|
|
|
|
def JumpToPreviewWindow():
|
|
|
|
""" Jump the vim cursor to the preview window, which must be active. Returns
|
|
|
|
boolean indicating if the cursor ended up in the preview window """
|
|
|
|
vim.command( 'silent! wincmd P' )
|
|
|
|
return vim.current.window.options[ 'previewwindow' ]
|
|
|
|
|
|
|
|
|
|
|
|
def JumpToPreviousWindow():
|
|
|
|
""" Jump the vim cursor to its previous window position """
|
|
|
|
vim.command( 'silent! wincmd p' )
|
|
|
|
|
|
|
|
|
2015-10-27 17:00:11 +01:00
|
|
|
def JumpToTab( tab_number ):
|
|
|
|
"""Jump to Vim tab with corresponding number """
|
|
|
|
vim.command( 'silent! tabn {0}'.format( tab_number ) )
|
|
|
|
|
|
|
|
|
2015-09-06 20:07:42 +01:00
|
|
|
def OpenFileInPreviewWindow( filename ):
|
|
|
|
""" Open the supplied filename in the preview window """
|
|
|
|
vim.command( 'silent! pedit! ' + filename )
|
|
|
|
|
|
|
|
|
|
|
|
def WriteToPreviewWindow( message ):
|
|
|
|
""" Display the supplied message in the preview window """
|
|
|
|
|
|
|
|
# This isn't something that comes naturally to Vim. Vim only wants to show
|
|
|
|
# tags and/or actual files in the preview window, so we have to hack it a
|
|
|
|
# little bit. We generate a temporary file name and "open" that, then write
|
|
|
|
# the data to it. We make sure the buffer can't be edited or saved. Other
|
|
|
|
# approaches include simply opening a split, but we want to take advantage of
|
|
|
|
# the existing Vim options for preview window height, etc.
|
|
|
|
|
|
|
|
ClosePreviewWindow()
|
|
|
|
|
|
|
|
OpenFileInPreviewWindow( vim.eval( 'tempname()' ) )
|
|
|
|
|
|
|
|
if JumpToPreviewWindow():
|
|
|
|
# We actually got to the preview window. By default the preview window can't
|
|
|
|
# be changed, so we make it writable, write to it, then make it read only
|
|
|
|
# again.
|
|
|
|
vim.current.buffer.options[ 'modifiable' ] = True
|
|
|
|
vim.current.buffer.options[ 'readonly' ] = False
|
|
|
|
|
|
|
|
vim.current.buffer[:] = message.splitlines()
|
|
|
|
|
|
|
|
vim.current.buffer.options[ 'buftype' ] = 'nofile'
|
|
|
|
vim.current.buffer.options[ 'swapfile' ] = False
|
|
|
|
vim.current.buffer.options[ 'modifiable' ] = False
|
|
|
|
vim.current.buffer.options[ 'readonly' ] = True
|
|
|
|
|
|
|
|
# We need to prevent closing the window causing a warning about unsaved
|
|
|
|
# file, so we pretend to Vim that the buffer has not been changed.
|
|
|
|
vim.current.buffer.options[ 'modified' ] = False
|
|
|
|
|
|
|
|
JumpToPreviousWindow()
|
|
|
|
else:
|
|
|
|
# We couldn't get to the preview window, but we still want to give the user
|
|
|
|
# the information we have. The only remaining option is to echo to the
|
|
|
|
# status area.
|
|
|
|
EchoText( message )
|
2015-10-27 17:00:11 +01:00
|
|
|
|
|
|
|
|
|
|
|
def CheckFilename( filename ):
|
|
|
|
"""Check if filename is openable."""
|
|
|
|
try:
|
|
|
|
open( filename ).close()
|
|
|
|
except TypeError:
|
|
|
|
raise RuntimeError( "'{0}' is not a valid filename".format( filename ) )
|
|
|
|
except IOError as error:
|
|
|
|
raise RuntimeError(
|
|
|
|
"filename '{0}' cannot be opened. {1}".format( filename, error ) )
|
|
|
|
|
|
|
|
|
2015-11-11 14:32:01 +01:00
|
|
|
def BufferIsVisibleForFilename( filename ):
|
2015-10-27 17:00:11 +01:00
|
|
|
"""Check if a buffer exists for a specific file."""
|
2015-11-11 14:32:01 +01:00
|
|
|
buffer_number = GetBufferNumberForFilename( filename, False )
|
|
|
|
return BufferIsVisible( buffer_number )
|
2015-10-27 17:00:11 +01:00
|
|
|
|
|
|
|
|
|
|
|
def CloseBuffersForFilename( filename ):
|
|
|
|
"""Close all buffers for a specific file."""
|
|
|
|
buffer_number = GetBufferNumberForFilename( filename, False )
|
|
|
|
while buffer_number is not -1:
|
|
|
|
vim.command( 'silent! bwipeout! {0}'.format( buffer_number ) )
|
2015-11-11 14:32:01 +01:00
|
|
|
new_buffer_number = GetBufferNumberForFilename( filename, False )
|
|
|
|
if buffer_number == new_buffer_number:
|
|
|
|
raise RuntimeError( "Buffer {0} for filename '{1}' should already be "
|
|
|
|
"wiped out.".format( buffer_number, filename ) )
|
|
|
|
buffer_number = new_buffer_number
|
2015-10-27 17:00:11 +01:00
|
|
|
|
|
|
|
|
|
|
|
def OpenFilename( filename, options = {} ):
|
|
|
|
"""Open a file in Vim. Following options are available:
|
|
|
|
- command: specify which Vim command is used to open the file. Choices
|
|
|
|
are same-buffer, horizontal-split, vertical-split, and new-tab (default:
|
|
|
|
horizontal-split);
|
|
|
|
- size: set the height of the window for a horizontal split or the width for
|
|
|
|
a vertical one (default: '');
|
|
|
|
- fix: set the winfixheight option for a horizontal split or winfixwidth for
|
|
|
|
a vertical one (default: False). See :h winfix for details;
|
|
|
|
- focus: focus the opened file (default: False);
|
|
|
|
- watch: automatically watch for changes (default: False). This is useful
|
|
|
|
for logs;
|
|
|
|
- position: set the position where the file is opened (default: start).
|
|
|
|
Choices are start and end."""
|
|
|
|
|
|
|
|
# Set the options.
|
|
|
|
command = GetVimCommand( options.get( 'command', 'horizontal-split' ),
|
|
|
|
'horizontal-split' )
|
|
|
|
size = ( options.get( 'size', '' ) if command in [ 'split', 'vsplit' ] else
|
|
|
|
'' )
|
|
|
|
focus = options.get( 'focus', False )
|
|
|
|
watch = options.get( 'watch', False )
|
|
|
|
position = options.get( 'position', 'start' )
|
|
|
|
|
|
|
|
# There is no command in Vim to return to the previous tab so we need to
|
|
|
|
# remember the current tab if needed.
|
|
|
|
if not focus and command is 'tabedit':
|
|
|
|
previous_tab = GetIntValue( 'tabpagenr()' )
|
|
|
|
|
|
|
|
# Open the file
|
|
|
|
CheckFilename( filename )
|
|
|
|
vim.command( 'silent! {0}{1} {2}'.format( size, command, filename ) )
|
|
|
|
|
|
|
|
if command is 'split':
|
|
|
|
vim.current.window.options[ 'winfixheight' ] = options.get( 'fix', False )
|
|
|
|
if command is 'vsplit':
|
|
|
|
vim.current.window.options[ 'winfixwidth' ] = options.get( 'fix', False )
|
|
|
|
|
|
|
|
if watch:
|
|
|
|
vim.current.buffer.options[ 'autoread' ] = True
|
|
|
|
vim.command( "exec 'au BufEnter <buffer> :silent! checktime {0}'"
|
|
|
|
.format( filename ) )
|
|
|
|
|
|
|
|
if position is 'end':
|
|
|
|
vim.command( 'silent! normal G zz' )
|
|
|
|
|
|
|
|
# Vim automatically set the focus to the opened file so we need to get the
|
|
|
|
# focus back (if the focus option is disabled) when opening a new tab or
|
|
|
|
# window.
|
|
|
|
if not focus:
|
|
|
|
if command is 'tabedit':
|
|
|
|
JumpToTab( previous_tab )
|
|
|
|
if command in [ 'split', 'vsplit' ]:
|
|
|
|
JumpToPreviousWindow()
|