YouCompleteMe/python/ycm/client/command_request.py

167 lines
5.8 KiB
Python
Raw Normal View History

# Copyright (C) 2013 Google Inc.
#
# 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/>.
2016-02-27 19:12:24 -05:00
from __future__ import unicode_literals
from __future__ import print_function
from __future__ import division
from __future__ import absolute_import
# Not installing aliases from python-future; it's unreliable and slow.
2016-02-27 19:12:24 -05:00
from builtins import * # noqa
2017-05-17 05:12:00 -04:00
from ycm.client.base_request import BaseRequest, BuildRequestData
2013-09-27 16:52:04 -04:00
from ycm import vimsupport
2016-02-19 14:02:58 -05:00
from ycmd.utils import ToUnicode
2015-09-12 10:30:21 -04:00
def _EnsureBackwardsCompatibility( arguments ):
if arguments and arguments[ 0 ] == 'GoToDefinitionElseDeclaration':
arguments[ 0 ] = 'GoTo'
return arguments
class CommandRequest( BaseRequest ):
def __init__( self,
arguments,
buffer_command = 'same-buffer',
extra_data = None ):
super( CommandRequest, self ).__init__()
self._arguments = _EnsureBackwardsCompatibility( arguments )
2018-02-07 13:57:45 -05:00
self._command = arguments and arguments[ 0 ]
self._buffer_command = buffer_command
self._extra_data = extra_data
2013-09-27 16:52:04 -04:00
self._response = None
def Start( self ):
request_data = BuildRequestData()
if self._extra_data:
request_data.update( self._extra_data )
request_data.update( {
'command_arguments': self._arguments
} )
2017-05-17 05:12:00 -04:00
self._response = self.PostDataToHandler( request_data,
'run_completer_command' )
def Response( self ):
2013-09-27 16:52:04 -04:00
return self._response
2015-09-12 10:30:21 -04:00
2018-07-24 11:12:50 -04:00
def RunPostCommandActionsIfNeeded( self, modifiers ):
if not self.Done() or self._response is None:
2013-09-27 16:52:04 -04:00
return
# If not a dictionary or a list, the response is necessarily a
# scalar: boolean, number, string, etc. In this case, we print
# it to the user.
if not isinstance( self._response, ( dict, list ) ):
return self._HandleBasicResponse()
if 'fixits' in self._response:
return self._HandleFixitResponse()
if 'message' in self._response:
return self._HandleMessageResponse()
if 'detailed_info' in self._response:
return self._HandleDetailedInfoResponse()
# The only other type of response we understand is GoTo, and that is the
# only one that we can't detect just by inspecting the response (it should
# either be a single location or a list)
2018-07-24 11:12:50 -04:00
return self._HandleGotoResponse( modifiers )
2015-09-12 10:30:21 -04:00
2018-07-24 11:12:50 -04:00
def _HandleGotoResponse( self, modifiers ):
if isinstance( self._response, list ):
vimsupport.SetQuickFixList(
[ _BuildQfListItem( x ) for x in self._response ] )
vimsupport.OpenQuickFixList( focus = True, autoclose = True )
else:
vimsupport.JumpToLocation( self._response[ 'filepath' ],
2015-09-12 10:30:21 -04:00
self._response[ 'line_num' ],
2018-07-24 11:12:50 -04:00
self._response[ 'column_num' ],
modifiers,
self._buffer_command )
2015-09-12 10:30:21 -04:00
def _HandleFixitResponse( self ):
if not len( self._response[ 'fixits' ] ):
vimsupport.PostVimMessage( 'No fixits found for current line',
warning = False )
else:
try:
fixit_index = 0
# When there are multiple fixit suggestions, present them as a list to
# the user hand have her choose which one to apply.
if len( self._response[ 'fixits' ] ) > 1:
fixit_index = vimsupport.SelectFromList(
"Multiple FixIt suggestions are available at this location. "
"Which one would you like to apply?",
[ fixit[ 'text' ] for fixit in self._response[ 'fixits' ] ] )
vimsupport.ReplaceChunks(
2018-02-07 13:57:45 -05:00
self._response[ 'fixits' ][ fixit_index ][ 'chunks' ],
silent = self._command == 'Format' )
except RuntimeError as e:
vimsupport.PostVimMessage( str( e ) )
2015-09-12 10:30:21 -04:00
2013-09-27 16:52:04 -04:00
def _HandleBasicResponse( self ):
vimsupport.PostVimMessage( self._response, warning = False )
def _HandleMessageResponse( self ):
vimsupport.PostVimMessage( self._response[ 'message' ], warning = False )
2013-09-27 16:52:04 -04:00
2015-09-12 10:30:21 -04:00
def _HandleDetailedInfoResponse( self ):
vimsupport.WriteToPreviewWindow( self._response[ 'detailed_info' ] )
def SendCommandRequest( arguments,
modifiers,
buffer_command,
extra_data = None ):
request = CommandRequest( arguments, buffer_command, extra_data )
# This is a blocking call.
request.Start()
2018-07-24 11:12:50 -04:00
request.RunPostCommandActionsIfNeeded( modifiers )
return request.Response()
2013-09-27 16:52:04 -04:00
def _BuildQfListItem( goto_data_item ):
qf_item = {}
if 'filepath' in goto_data_item:
2016-02-19 14:02:58 -05:00
qf_item[ 'filename' ] = ToUnicode( goto_data_item[ 'filepath' ] )
2013-09-27 16:52:04 -04:00
if 'description' in goto_data_item:
2016-02-19 14:02:58 -05:00
qf_item[ 'text' ] = ToUnicode( goto_data_item[ 'description' ] )
2013-09-27 16:52:04 -04:00
if 'line_num' in goto_data_item:
qf_item[ 'lnum' ] = goto_data_item[ 'line_num' ]
2013-09-27 16:52:04 -04:00
if 'column_num' in goto_data_item:
# ycmd returns columns 1-based, and QuickFix lists require "byte offsets".
# See :help getqflist and equivalent comment in
# vimsupport.ConvertDiagnosticsToQfList.
#
# When the Vim help says "byte index", it really means "1-based column
# number" (which is somewhat confusing). :help getqflist states "first
# column is 1".
qf_item[ 'col' ] = goto_data_item[ 'column_num' ]
2013-09-27 16:52:04 -04:00
return qf_item