mirror of
https://github.com/apple/swift.git
synced 2025-12-14 20:36:38 +01:00
When completing in the only expression of closure, use the return type of the closure as the type context for the code-completion. However, since code-completion may be on an incomplete input, we only use the return type to improve the quality of the result, not to mark it invalid, since (a) we may add another statement afterwards, or (b) if the context type is Void it doesn't need to match the value.
247 lines
8.5 KiB
C++
247 lines
8.5 KiB
C++
//===--- CodeCompletionCallbacks.h - Parser's interface to code completion ===//
|
|
//
|
|
// This source file is part of the Swift.org open source project
|
|
//
|
|
// Copyright (c) 2014 - 2018 Apple Inc. and the Swift project authors
|
|
// Licensed under Apache License v2.0 with Runtime Library Exception
|
|
//
|
|
// See https://swift.org/LICENSE.txt for license information
|
|
// See https://swift.org/CONTRIBUTORS.txt for the list of Swift project authors
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#ifndef SWIFT_PARSE_CODE_COMPLETION_CALLBACKS_H
|
|
#define SWIFT_PARSE_CODE_COMPLETION_CALLBACKS_H
|
|
|
|
#include "swift/AST/ASTContext.h"
|
|
#include "swift/Parse/Parser.h"
|
|
|
|
namespace swift {
|
|
|
|
enum class ObjCSelectorContext {
|
|
/// Code completion is not performed inside #selector
|
|
None,
|
|
/// Code completion is performed in a method #selector
|
|
MethodSelector,
|
|
/// Code completion is performed inside #selector(getter:)
|
|
GetterSelector,
|
|
/// Code completion is performed inside #selector(setter:)
|
|
SetterSelector
|
|
};
|
|
|
|
/// Parser's interface to code completion.
|
|
class CodeCompletionCallbacks {
|
|
protected:
|
|
Parser &P;
|
|
ASTContext &Context;
|
|
ParserPosition ExprBeginPosition;
|
|
|
|
/// The declaration parsed during delayed parsing that was caused by code
|
|
/// completion. This declaration contained the code completion token.
|
|
Decl *ParsedDecl = nullptr;
|
|
|
|
/// True if code completion is done inside a raw value expression of an enum
|
|
/// case.
|
|
bool InEnumElementRawValue = false;
|
|
|
|
/// Whether or not the expression that is currently parsed is inside a
|
|
/// \c #selector and if so, which kind of selector
|
|
ObjCSelectorContext ParseExprSelectorContext = ObjCSelectorContext::None;
|
|
|
|
/// Whether or not the expression that shall be completed is inside a
|
|
/// \c #selector and if so, which kind of selector
|
|
ObjCSelectorContext CompleteExprSelectorContext = ObjCSelectorContext::None;
|
|
|
|
std::vector<Expr *> leadingSequenceExprs;
|
|
|
|
public:
|
|
CodeCompletionCallbacks(Parser &P)
|
|
: P(P), Context(P.Context) {
|
|
}
|
|
|
|
virtual ~CodeCompletionCallbacks() {}
|
|
|
|
bool isInsideObjCSelector() const {
|
|
return CompleteExprSelectorContext != ObjCSelectorContext::None;
|
|
}
|
|
|
|
void setExprBeginning(ParserPosition PP) {
|
|
ExprBeginPosition = PP;
|
|
}
|
|
|
|
/// Set the decl inside which the code-completion occurred. This is used when
|
|
/// completing inside a parameter list or where clause where the Parser's
|
|
/// CurDeclContext will not be where we want to perform lookup.
|
|
void setParsedDecl(Decl *D) {
|
|
ParsedDecl = D;
|
|
}
|
|
|
|
void setLeadingSequenceExprs(ArrayRef<Expr *> exprs) {
|
|
leadingSequenceExprs.assign(exprs.begin(), exprs.end());
|
|
}
|
|
|
|
class InEnumElementRawValueRAII {
|
|
CodeCompletionCallbacks *Callbacks;
|
|
|
|
public:
|
|
InEnumElementRawValueRAII(CodeCompletionCallbacks *Callbacks)
|
|
: Callbacks(Callbacks) {
|
|
if (Callbacks)
|
|
Callbacks->InEnumElementRawValue = true;
|
|
}
|
|
|
|
~InEnumElementRawValueRAII() {
|
|
if (Callbacks)
|
|
Callbacks->InEnumElementRawValue = false;
|
|
}
|
|
};
|
|
|
|
/// RAII type that temporarily sets the "in Objective-C #selector expression"
|
|
/// flag on the code completion callbacks object.
|
|
class InObjCSelectorExprRAII {
|
|
CodeCompletionCallbacks *Callbacks;
|
|
|
|
public:
|
|
InObjCSelectorExprRAII(CodeCompletionCallbacks *Callbacks,
|
|
ObjCSelectorContext SelectorContext)
|
|
: Callbacks(Callbacks) {
|
|
if (Callbacks)
|
|
Callbacks->ParseExprSelectorContext = SelectorContext;
|
|
}
|
|
|
|
~InObjCSelectorExprRAII() {
|
|
if (Callbacks)
|
|
Callbacks->ParseExprSelectorContext = ObjCSelectorContext::None;
|
|
}
|
|
};
|
|
|
|
/// Complete the whole expression. This is a fallback that should
|
|
/// produce results when more specific completion methods failed.
|
|
virtual void completeExpr() {};
|
|
|
|
/// Complete expr-dot after we have consumed the dot.
|
|
virtual void completeDotExpr(Expr *E, SourceLoc DotLoc) {};
|
|
|
|
/// Complete the beginning of a statement or expression.
|
|
virtual void completeStmtOrExpr(CodeCompletionExpr *E) {};
|
|
|
|
/// Complete the beginning of expr-postfix -- no tokens provided
|
|
/// by user.
|
|
virtual void completePostfixExprBeginning(CodeCompletionExpr *E) {};
|
|
|
|
/// Complete the beginning of expr-postfix in a for-each loop sequqence
|
|
/// -- no tokens provided by user.
|
|
virtual void completeForEachSequenceBeginning(CodeCompletionExpr *E) {};
|
|
|
|
/// Complete a given expr-postfix.
|
|
virtual void completePostfixExpr(Expr *E, bool hasSpace) {};
|
|
|
|
/// Complete a given expr-postfix, given that there is a following
|
|
/// left parenthesis.
|
|
virtual void completePostfixExprParen(Expr *E, Expr *CodeCompletionE) {};
|
|
|
|
/// Complete expr-super after we have consumed the 'super' keyword.
|
|
virtual void completeExprSuper(SuperRefExpr *SRE) {};
|
|
|
|
/// Complete expr-super after we have consumed the 'super' keyword and
|
|
/// a dot.
|
|
virtual void completeExprSuperDot(SuperRefExpr *SRE) {};
|
|
|
|
/// Complete the argument to an Objective-C #keyPath
|
|
/// expression.
|
|
///
|
|
/// \param KPE A partial #keyPath expression that can be used to
|
|
/// provide context. This will be \c NULL if no components of the
|
|
/// #keyPath argument have been parsed yet.
|
|
virtual void completeExprKeyPath(KeyPathExpr *KPE, SourceLoc DotLoc) {};
|
|
|
|
/// Complete the beginning of type-simple -- no tokens provided
|
|
/// by user.
|
|
virtual void completeTypeSimpleBeginning() {};
|
|
|
|
/// Complete a given type-identifier after we have consumed the dot.
|
|
virtual void completeTypeIdentifierWithDot(IdentTypeRepr *ITR) {};
|
|
|
|
/// Complete a given type-identifier when there is no trailing dot.
|
|
virtual void completeTypeIdentifierWithoutDot(IdentTypeRepr *ITR) {};
|
|
|
|
/// Complete at the beginning of a case stmt pattern.
|
|
virtual void completeCaseStmtBeginning() {};
|
|
|
|
/// Complete a case stmt pattern that starts with a dot.
|
|
virtual void completeCaseStmtDotPrefix() {};
|
|
|
|
/// Complete at the beginning of member of a nominal decl member -- no tokens
|
|
/// provided by user.
|
|
virtual void completeNominalMemberBeginning(
|
|
SmallVectorImpl<StringRef> &Keywords, SourceLoc introducerLoc) {};
|
|
|
|
/// Complete at the beginning of accessor in a accessor block.
|
|
virtual void completeAccessorBeginning() {};
|
|
|
|
/// Complete the keyword in attribute, for instance, @available.
|
|
virtual void completeDeclAttrKeyword(Decl *D, bool Sil, bool Param) {};
|
|
|
|
/// Complete the parameters in attribute, for instance, version specifier for
|
|
/// @available.
|
|
virtual void completeDeclAttrParam(DeclAttrKind DK, int Index) {};
|
|
|
|
/// Complete within a precedence group decl or after a colon in an
|
|
/// operator decl.
|
|
virtual void completeInPrecedenceGroup(SyntaxKind SK) {};
|
|
|
|
/// Complete the platform names inside #available statements.
|
|
virtual void completePoundAvailablePlatform() {};
|
|
|
|
/// Complete the import decl with importable modules.
|
|
virtual void
|
|
completeImportDecl(std::vector<std::pair<Identifier, SourceLoc>> &Path) {};
|
|
|
|
/// Complete unresolved members after dot.
|
|
virtual void completeUnresolvedMember(CodeCompletionExpr *E,
|
|
SourceLoc DotLoc) {};
|
|
|
|
virtual void completeAssignmentRHS(AssignExpr *E) {};
|
|
|
|
virtual void completeCallArg(CodeCompletionExpr *E) {};
|
|
|
|
virtual void completeReturnStmt(CodeCompletionExpr *E) {};
|
|
|
|
/// Complete a yield statement. A missing yield index means that the
|
|
/// completion immediately follows the 'yield' keyword; it may be either
|
|
/// an expresion or a parenthesized expression list. A present yield
|
|
/// index means that the completion is within the parentheses and is
|
|
/// for a specific yield value.
|
|
virtual void completeYieldStmt(CodeCompletionExpr *E,
|
|
Optional<unsigned> yieldIndex) {};
|
|
|
|
virtual void completeAfterPoundExpr(CodeCompletionExpr *E,
|
|
Optional<StmtKind> ParentKind) {};
|
|
|
|
virtual void completeAfterPoundDirective() {};
|
|
|
|
virtual void completePlatformCondition() {};
|
|
|
|
virtual void completeAfterIfStmt(bool hasElse) {};
|
|
|
|
virtual void completeGenericParams(TypeLoc TL) {};
|
|
|
|
/// Signals that the AST for the all the delayed-parsed code was
|
|
/// constructed. No \c complete*() callbacks will be done after this.
|
|
virtual void doneParsing() = 0;
|
|
};
|
|
|
|
/// A factory to create instances of \c CodeCompletionCallbacks.
|
|
class CodeCompletionCallbacksFactory {
|
|
public:
|
|
virtual ~CodeCompletionCallbacksFactory() {}
|
|
|
|
/// Create an instance of \c CodeCompletionCallbacks. The result
|
|
/// should be deallocated with 'delete'.
|
|
virtual CodeCompletionCallbacks *createCodeCompletionCallbacks(Parser &P) = 0;
|
|
};
|
|
|
|
} // namespace swift
|
|
|
|
#endif // LLVM_SWIFT_PARSE_CODE_COMPLETION_CALLBACKS_H
|