1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
|
local source = {}
source.new = function(client)
local self = setmetatable({}, { __index = source })
self.client = client
self.request_ids = {}
return self
end
source.get_debug_name = function(self)
return table.concat({ 'nvim_lsp', self.client.name }, ':')
end
source.is_available = function(self)
-- client is stopped.
if self.client.is_stopped() then
return false
end
-- client is not attached to current buffer.
if not vim.lsp.buf_get_clients(vim.api.nvim_get_current_buf())[self.client.id] then
return false
end
-- client has no completion capability.
if not self:_get(self.client.server_capabilities, { 'completionProvider' }) then
return false
end
return true;
end
source.get_trigger_characters = function(self)
return self:_get(self.client.server_capabilities, { 'completionProvider', 'triggerCharacters' }) or {}
end
source.complete = function(self, request, callback)
local params = vim.lsp.util.make_position_params(0, self.client.offset_encoding)
params.context = {}
params.context.triggerKind = request.completion_context.triggerKind
params.context.triggerCharacter = request.completion_context.triggerCharacter
self:_request('textDocument/completion', params, function(_, response)
callback(response)
end)
end
source.resolve = function(self, completion_item, callback)
-- client is stopped.
if self.client.is_stopped() then
return callback()
end
-- client has no completion capability.
if not self:_get(self.client.server_capabilities, { 'completionProvider', 'resolveProvider' }) then
return callback()
end
self:_request('completionItem/resolve', completion_item, function(_, response)
callback(response or completion_item)
end)
end
source.execute = function(self, completion_item, callback)
-- client is stopped.
if self.client.is_stopped() then
return callback()
end
-- completion_item has no command.
if not completion_item.command then
return callback()
end
self:_request('workspace/executeCommand', completion_item.command, function(_, _)
callback()
end)
end
source._get = function(_, root, paths)
local c = root
for _, path in ipairs(paths) do
c = c[path]
if not c then
return nil
end
end
return c
end
source._request = function(self, method, params, callback)
if self.request_ids[method] ~= nil then
self.client.cancel_request(self.request_ids[method])
self.request_ids[method] = nil
end
local _, request_id
_, request_id = self.client.request(method, params, function(arg1, arg2, arg3)
if self.request_ids[method] ~= request_id then
return
end
self.request_ids[method] = nil
-- Text changed, retry
if arg1 and arg1.code == -32801 then
self:_request(method, params, callback)
return
end
if method == arg2 then
callback(arg1, arg3) -- old signature
else
callback(arg1, arg2) -- new signature
end
end)
self.request_ids[method] = request_id
end
return source
|