summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--changelog.md1
-rw-r--r--script/parser/guide.lua130
-rw-r--r--test/hover/init.lua13
3 files changed, 96 insertions, 48 deletions
diff --git a/changelog.md b/changelog.md
index ad7b87a6..1f6e3195 100644
--- a/changelog.md
+++ b/changelog.md
@@ -2,6 +2,7 @@
## 1.5.0
* `NEW` setting `runtime.unicodeName`
+* `NEW` fully supports `---@generic T`
* `FIX` [#274](https://github.com/sumneko/lua-language-server/issues/274)
## 1.4.0
diff --git a/script/parser/guide.lua b/script/parser/guide.lua
index 7df247f2..19e60a0e 100644
--- a/script/parser/guide.lua
+++ b/script/parser/guide.lua
@@ -1418,7 +1418,41 @@ function m.checkSameSimpleInSpecialBranch(status, obj, start, queue)
end
end
-function m.checkSameSimpleByDocType(status, doc)
+local function stepRefOfGeneric(status, typeUnit, args, mode)
+ if not args then
+ return nil
+ end
+ local results = {}
+ local myName = typeUnit[1]
+ for _, typeName in ipairs(typeUnit.typeGeneric[myName]) do
+ if typeName == typeUnit then
+ goto CONTINUE
+ end
+ local doc = m.getDocState(typeName)
+ if doc.type ~= 'doc.param' then
+ goto CONTINUE
+ end
+ if not doc.bindSources then
+ goto CONTINUE
+ end
+ local paramName = doc.param[1]
+ for _, source in ipairs(doc.bindSources) do
+ if source.type == 'local'
+ and source[1] == paramName
+ and source.parent.type == 'funcargs' then
+ for index, arg in ipairs(source.parent) do
+ if arg == source then
+ results[#results+1] = args[index]
+ end
+ end
+ end
+ end
+ ::CONTINUE::
+ end
+ return results
+end
+
+function m.checkSameSimpleByDocType(status, doc, args)
if status.cache.searchingBindedDoc then
return
end
@@ -1427,9 +1461,16 @@ function m.checkSameSimpleByDocType(status, doc)
end
local results = {}
for _, piece in ipairs(doc.types) do
- local pieceResult = stepRefOfDocType(status, piece, 'def')
- for _, res in ipairs(pieceResult) do
- results[#results+1] = res
+ if piece.typeGeneric then
+ local pieceResult = stepRefOfGeneric(status, piece, args, 'def')
+ for _, res in ipairs(pieceResult) do
+ results[#results+1] = res
+ end
+ else
+ local pieceResult = stepRefOfDocType(status, piece, 'def')
+ for _, res in ipairs(pieceResult) do
+ results[#results+1] = res
+ end
end
end
return results
@@ -1703,7 +1744,7 @@ function m.searchSameFieldsCrossMethod(status, ref, start, queue)
end
end
-local function checkSameSimpleAndMergeFunctionReturnsByDoc(status, results, source, index, call)
+local function checkSameSimpleAndMergeFunctionReturnsByDoc(status, results, source, index, args)
if not source or source.type ~= 'function' then
return
end
@@ -1722,7 +1763,7 @@ local function checkSameSimpleAndMergeFunctionReturnsByDoc(status, results, sour
if not rtn then
return
end
- local types = m.checkSameSimpleByDocType(status, rtn)
+ local types = m.checkSameSimpleByDocType(status, rtn, args)
if not types then
return
end
@@ -1765,7 +1806,7 @@ function m.checkSameSimpleInCallInSameFile(status, func, args, index)
m.searchRefs(newStatus, func, 'def')
for _, def in ipairs(newStatus.results) do
local hasDocReturn = checkSameSimpleAndMergeDocTypeFunctionReturns(status, results, def, index)
- or checkSameSimpleAndMergeFunctionReturnsByDoc(status, results, def, index)
+ or checkSameSimpleAndMergeFunctionReturnsByDoc(status, results, def, index, args)
if not hasDocReturn then
local value = m.getObjectValue(def) or def
if value.type == 'function' then
@@ -2157,6 +2198,11 @@ function m.pushResult(status, mode, ref, simple)
results[#results+1] = ref
end
end
+ if m.isLiteral(ref)
+ and ref.parent.type == 'callargs'
+ and ref ~= simple.node then
+ results[#results+1] = ref
+ end
elseif mode == 'ref' then
if ref.type == 'setfield'
or ref.type == 'getfield'
@@ -2192,6 +2238,11 @@ function m.pushResult(status, mode, ref, simple)
if ref.parent and ref.parent.type == 'return' then
results[#results+1] = ref
end
+ if m.isLiteral(ref)
+ and ref.parent.type == 'callargs'
+ and ref ~= simple.node then
+ results[#results+1] = ref
+ end
elseif mode == 'field' then
if ref.type == 'setfield'
or ref.type == 'getfield'
@@ -2644,6 +2695,7 @@ function m.viewInferType(infers)
local mark = {}
local types = {}
local hasDoc
+ local generic
for i = 1, #infers do
local infer = infers[i]
local src = infer.source
@@ -2652,7 +2704,10 @@ function m.viewInferType(infers)
or src.type == 'doc.type.name'
or src.type == 'doc.type.array'
or src.type == 'doc.type.generic' then
- if infer.type ~= 'any' then
+ if src.typeGeneric then
+ generic = infer
+ end
+ if infer.type ~= 'any' and not src.typeGeneric then
hasDoc = true
break
end
@@ -2678,13 +2733,21 @@ function m.viewInferType(infers)
end
else
for i = 1, #infers do
- local tp = infers[i].type or 'any'
+ local infer = infers[i]
+ if infer.source.typeGeneric then
+ goto CONTINUE
+ end
+ local tp = infer.type or 'any'
if not mark[tp] then
types[#types+1] = tp
end
mark[tp] = true
+ ::CONTINUE::
end
end
+ if #types == 0 and generic then
+ types[#types+1] = generic.type
+ end
return m.mergeTypes(types)
end
@@ -2882,33 +2945,28 @@ local function getDocAliasExtends(status, typeUnit)
return nil
end
-local function getDocTypeUnitName(status, unit, genericCallback)
+local function getDocTypeUnitName(status, unit)
local typeName
if unit.type == 'doc.type.name' then
typeName = unit[1]
elseif unit.type == 'doc.type.function' then
typeName = 'function'
elseif unit.type == 'doc.type.array' then
- typeName = getDocTypeUnitName(status, unit.node, genericCallback) .. '[]'
+ typeName = getDocTypeUnitName(status, unit.node) .. '[]'
elseif unit.type == 'doc.type.generic' then
typeName = ('%s<%s, %s>'):format(
- getDocTypeUnitName(status, unit.node, genericCallback),
- m.viewInferType(m.getDocTypeNames(status, unit.key, genericCallback)),
- m.viewInferType(m.getDocTypeNames(status, unit.value, genericCallback))
+ getDocTypeUnitName(status, unit.node),
+ m.viewInferType(m.getDocTypeNames(status, unit.key)),
+ m.viewInferType(m.getDocTypeNames(status, unit.value))
)
end
if unit.typeGeneric then
- if genericCallback then
- typeName = genericCallback(typeName, unit)
- or ('<%s>'):format(typeName)
- else
- typeName = ('<%s>'):format(typeName)
- end
+ typeName = ('<%s>'):format(typeName)
end
return typeName
end
-function m.getDocTypeNames(status, doc, genericCallback)
+function m.getDocTypeNames(status, doc)
local results = {}
if not doc then
return results
@@ -2916,12 +2974,12 @@ function m.getDocTypeNames(status, doc, genericCallback)
for _, unit in ipairs(doc.types) do
local alias = getDocAliasExtends(status, unit)
if alias then
- local aliasResults = m.getDocTypeNames(status, alias, genericCallback)
+ local aliasResults = m.getDocTypeNames(status, alias)
for _, res in ipairs(aliasResults) do
results[#results+1] = res
end
else
- local typeName = getDocTypeUnitName(status, unit, genericCallback)
+ local typeName = getDocTypeUnitName(status, unit)
results[#results+1] = {
type = typeName,
source = unit,
@@ -3673,31 +3731,7 @@ local function mergeFunctionReturnsByDoc(status, source, index, call)
if not rtn then
return
end
- local results = m.getDocTypeNames(status, rtn, function (typeName, typeUnit)
- if not source.args or not call.args then
- return
- end
- local name = typeUnit[1]
- local generics = typeUnit.typeGeneric[name]
- if not generics then
- return
- end
- local first = generics[1]
- if not first or first == typeUnit then
- return
- end
- local docParam = m.getParentType(first, 'doc.param')
- local paramName = docParam.param[1]
- for i, arg in ipairs(source.args) do
- if arg[1] == paramName then
- local callArg = call.args[i]
- if not callArg then
- return
- end
- return m.viewInferType(m.searchInfer(status, callArg))
- end
- end
- end)
+ local results = m.getDocTypeNames(status, rtn)
if #results == 0 then
return
end
diff --git a/test/hover/init.lua b/test/hover/init.lua
index cc2fcea8..f70c8a05 100644
--- a/test/hover/init.lua
+++ b/test/hover/init.lua
@@ -1476,3 +1476,16 @@ local v: Object {
a: string,
}
]]
+
+--TEST [[
+-----@class C
+-----@field x string
+--local t
+--
+--local <?v?> = assert(t)
+--]]
+--[[
+--local v: C {
+-- x: string,
+--}
+--]]