mirror of
https://github.com/golang/go
synced 2024-11-18 18:54:42 -07:00
2047c2d578
Offer "struct", "interface", "map", "chan", and "func" keywords when we expect a type. For example "var foo i<>" will offer "interface". Because "struct" and "interface" are more often used when declaring named types, they get a higher score in type declarations. Otherwise, "map", "chan" and "func" get a higher score. I also got rid of the special keyword scoring. Now keywords just use stdScore and highScore. This makes the interplay with other types of candidates more predictable. Keywords are offered in pretty limited contexts, so I don't think they will be annoying. Finally, keyword candidate score is now to be scaled properly based on how well they match the prefix. Previously they weren't penalized for not matching well, so there were probably some situations where keywords were ranked too high. Updates golang/go#34009. Change-Id: I0b659c00a8503cd72da28853dfe54fcb67f734ae Reviewed-on: https://go-review.googlesource.com/c/tools/+/220503 Run-TryBot: Muir Manders <muir@mnd.rs> TryBot-Result: Gobot Gobot <gobot@golang.org> Reviewed-by: Rebecca Stambler <rstambler@golang.org>
132 lines
3.6 KiB
Go
132 lines
3.6 KiB
Go
package source
|
|
|
|
import (
|
|
"go/ast"
|
|
|
|
"golang.org/x/tools/internal/lsp/protocol"
|
|
)
|
|
|
|
const (
|
|
BREAK = "break"
|
|
CASE = "case"
|
|
CHAN = "chan"
|
|
CONST = "const"
|
|
CONTINUE = "continue"
|
|
DEFAULT = "default"
|
|
DEFER = "defer"
|
|
ELSE = "else"
|
|
FALLTHROUGH = "fallthrough"
|
|
FOR = "for"
|
|
FUNC = "func"
|
|
GO = "go"
|
|
GOTO = "goto"
|
|
IF = "if"
|
|
IMPORT = "import"
|
|
INTERFACE = "interface"
|
|
MAP = "map"
|
|
PACKAGE = "package"
|
|
RANGE = "range"
|
|
RETURN = "return"
|
|
SELECT = "select"
|
|
STRUCT = "struct"
|
|
SWITCH = "switch"
|
|
TYPE = "type"
|
|
VAR = "var"
|
|
)
|
|
|
|
// addKeywordCompletions offers keyword candidates appropriate at the position.
|
|
func (c *completer) addKeywordCompletions() {
|
|
seen := make(map[string]bool)
|
|
|
|
// addKeywords dedupes and adds completion items for the specified
|
|
// keywords with the specified score.
|
|
addKeywords := func(score float64, kws ...string) {
|
|
for _, kw := range kws {
|
|
if seen[kw] {
|
|
continue
|
|
}
|
|
seen[kw] = true
|
|
|
|
if matchScore := c.matcher.Score(kw); matchScore > 0 {
|
|
c.items = append(c.items, CompletionItem{
|
|
Label: kw,
|
|
Kind: protocol.KeywordCompletion,
|
|
InsertText: kw,
|
|
Score: score * float64(matchScore),
|
|
})
|
|
}
|
|
}
|
|
}
|
|
|
|
if c.wantTypeName() {
|
|
// If we expect a type name, include "interface", "struct",
|
|
// "func", "chan", and "map".
|
|
|
|
// "interface" and "struct" are more common declaring named types.
|
|
// Give them a higher score if we are in a type declaration.
|
|
structIntf, funcChanMap := stdScore, highScore
|
|
if len(c.path) > 1 {
|
|
if _, namedDecl := c.path[1].(*ast.TypeSpec); namedDecl {
|
|
structIntf, funcChanMap = highScore, stdScore
|
|
}
|
|
}
|
|
|
|
addKeywords(structIntf, STRUCT, INTERFACE)
|
|
addKeywords(funcChanMap, FUNC, CHAN, MAP)
|
|
}
|
|
|
|
// If we are at the file scope, only offer decl keywords. We don't
|
|
// get *ast.Idents at the file scope because non-keyword identifiers
|
|
// turn into *ast.BadDecl, not *ast.Ident.
|
|
if len(c.path) == 1 || isASTFile(c.path[1]) {
|
|
addKeywords(stdScore, TYPE, CONST, VAR, FUNC, IMPORT)
|
|
return
|
|
} else if _, ok := c.path[0].(*ast.Ident); !ok {
|
|
// Otherwise only offer keywords if the client is completing an identifier.
|
|
return
|
|
}
|
|
|
|
// Only suggest keywords if we are beginning a statement.
|
|
switch c.path[1].(type) {
|
|
case *ast.BlockStmt, *ast.CommClause, *ast.CaseClause, *ast.ExprStmt:
|
|
default:
|
|
return
|
|
}
|
|
|
|
// Filter out keywords depending on scope
|
|
// Skip the first one because we want to look at the enclosing scopes
|
|
path := c.path[1:]
|
|
for i, n := range path {
|
|
switch node := n.(type) {
|
|
case *ast.CaseClause:
|
|
// only recommend "fallthrough" and "break" within the bodies of a case clause
|
|
if c.pos > node.Colon {
|
|
addKeywords(stdScore, BREAK)
|
|
// "fallthrough" is only valid in switch statements.
|
|
// A case clause is always nested within a block statement in a switch statement,
|
|
// that block statement is nested within either a TypeSwitchStmt or a SwitchStmt.
|
|
if i+2 >= len(path) {
|
|
continue
|
|
}
|
|
if _, ok := path[i+2].(*ast.SwitchStmt); ok {
|
|
addKeywords(stdScore, FALLTHROUGH)
|
|
}
|
|
}
|
|
case *ast.CommClause:
|
|
if c.pos > node.Colon {
|
|
addKeywords(stdScore, BREAK)
|
|
}
|
|
case *ast.TypeSwitchStmt, *ast.SelectStmt, *ast.SwitchStmt:
|
|
addKeywords(stdScore, CASE, DEFAULT)
|
|
case *ast.ForStmt:
|
|
addKeywords(stdScore, BREAK, CONTINUE)
|
|
// This is a bit weak, functions allow for many keywords
|
|
case *ast.FuncDecl:
|
|
if node.Body != nil && c.pos > node.Body.Lbrace {
|
|
addKeywords(stdScore, DEFER, RETURN, FOR, GO, SWITCH, SELECT, IF, ELSE, VAR, CONST, GOTO, TYPE)
|
|
}
|
|
}
|
|
}
|
|
|
|
}
|