summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
author最萌小汐 <sumneko@hotmail.com>2019-03-12 16:03:29 +0800
committer最萌小汐 <sumneko@hotmail.com>2019-03-12 16:03:29 +0800
commit26acef86a9acef06f63605a2dfff0647f36655de (patch)
treefd7652efe208ef2200d5c9f26e6e6ac08714fc31
parent629c630ff23fd4d4eea498ba2e0984f0751e3a17 (diff)
downloadlua-language-server-26acef86a9acef06f63605a2dfff0647f36655de.zip
改进相似性实现
-rw-r--r--server/src/core/definition.lua10
-rw-r--r--server/src/core/implementation.lua119
2 files changed, 105 insertions, 24 deletions
diff --git a/server/src/core/definition.lua b/server/src/core/definition.lua
index 9b9cd75e..35dfc03b 100644
--- a/server/src/core/definition.lua
+++ b/server/src/core/definition.lua
@@ -6,14 +6,20 @@ local function parseValueSimily(vm, source, lsp)
local positions = {}
for _, other in ipairs(vm.sources) do
if other == source then
- break
+ goto CONTINUE
end
- if other[1] == key and not other:bindLocal() and other:bindValue() and other:action() == 'set' then
+ if other[1] == key
+ and not other:bindLocal()
+ and other:bindValue()
+ and other:action() == 'set'
+ and source:bindValue() ~= other:bindValue()
+ then
positions[#positions+1] = {
other.start,
other.finish,
}
end
+ :: CONTINUE ::
end
if #positions == 0 then
return nil
diff --git a/server/src/core/implementation.lua b/server/src/core/implementation.lua
index 1d36bd4d..b7ca2c6e 100644
--- a/server/src/core/implementation.lua
+++ b/server/src/core/implementation.lua
@@ -1,48 +1,111 @@
-local function parseLocal(vm, loc, lsp)
+local function parseValueSimily(vm, source, lsp)
+ local key = source[1]
+ if not key then
+ return nil
+ end
local positions = {}
- positions[#positions+1] = {
- loc.source.start,
- loc.source.finish,
- }
+ for _, other in ipairs(vm.sources) do
+ if other == source then
+ goto CONTINUE
+ end
+ if other[1] == key
+ and not other:bindLocal()
+ and other:bindValue()
+ and other:action() == 'set'
+ and source:bindValue() ~= other:bindValue()
+ then
+ positions[#positions+1] = {
+ other.start,
+ other.finish,
+ }
+ end
+ :: CONTINUE ::
+ end
if #positions == 0 then
return nil
end
return positions
end
-local function parseValue(vm, value, lsp)
+local function parseValueCrossFile(vm, source, lsp)
+ local value = source:bindValue()
local positions = {}
value:eachInfo(function (info)
- if info.type == 'set' then
+ if info.type == 'local' and info.source.uri == value.uri then
positions[#positions+1] = {
info.source.start,
info.source.finish,
+ value.uri,
}
+ return true
end
end)
- if #positions == 0 then
- return nil
+ if #positions > 0 then
+ return positions
+ end
+
+ value:eachInfo(function (info)
+ if info.type == 'set' and info.source.uri == value.uri then
+ positions[#positions+1] = {
+ info.source.start,
+ info.source.finish,
+ value.uri,
+ }
+ end
+ end)
+ if #positions > 0 then
+ return positions
+ end
+
+ value:eachInfo(function (info)
+ if info.type == 'return' and info.source.uri == value.uri then
+ positions[#positions+1] = {
+ info.source.start,
+ info.source.finish,
+ value.uri,
+ }
+ end
+ end)
+ if #positions > 0 then
+ return positions
+ end
+
+ local destVM = lsp:getVM(value.uri)
+ if not destVM then
+ positions[#positions+1] = {
+ 0, 0, value.uri,
+ }
+ return positions
+ end
+
+ local result = parseValueSimily(destVM, source, lsp)
+ if result then
+ for _, position in ipairs(result) do
+ positions[#positions+1] = position
+ position[3] = value.uri
+ end
end
+ if #positions > 0 then
+ return positions
+ end
+
return positions
end
-local function parseValueSimily(vm, source, lsp)
- local key = source[1]
- if not key then
- return nil
+local function parseValue(vm, source, lsp)
+ local value = source:bindValue()
+ if value.uri ~= vm.uri then
+ return parseValueCrossFile(vm, source, lsp)
end
local positions = {}
- for _, other in ipairs(vm.sources) do
- if other == source then
- break
- end
- if other[1] == key and not other:bindLocal() and other:bindValue() and other:action() == 'set' then
+ value:eachInfo(function (info)
+ if info.type == 'set' then
positions[#positions+1] = {
- other.start,
- other.finish,
+ info.source.start,
+ info.source.finish,
}
end
- end
+ end)
if #positions == 0 then
return nil
end
@@ -65,15 +128,27 @@ local function parseLabel(vm, label, lsp)
return positions
end
+local function jumpUri(vm, source, lsp)
+ local uri = source:get 'target uri'
+ local positions = {}
+ positions[#positions+1] = {
+ 0, 0, uri,
+ }
+ return positions
+end
+
return function (vm, source, lsp)
if not source then
return nil
end
if source:bindValue() then
- return parseValue(vm, source:bindValue(), lsp)
+ return parseValue(vm, source, lsp)
or parseValueSimily(vm, source, lsp)
end
if source:bindLabel() then
return parseLabel(vm, source:bindLabel(), lsp)
end
+ if source:get 'target uri' then
+ return jumpUri(vm, source, lsp)
+ end
end