Files
swift-mirror/lib/IDE/KeyPathCompletion.cpp
Alex Hoppen 72cadecf21 [CodeCompletion] Split result delivery into a result colleciton and consumer phase
This will allow us to run two different completion kinds and deliver results from both of them.

Also: Compute a unified type context for global lookup. Previously, we always used the expected type context of the last lookup. But really, we should be considering all possible types from all constraint system solutions when computing code completion results from the cache.
2023-08-24 15:41:36 -07:00

96 lines
3.4 KiB
C++

//===--- KeyPathCompletion.cpp --------------------------------------------===//
//
// This source file is part of the Swift.org open source project
//
// Copyright (c) 2014 - 2022 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
//
//===----------------------------------------------------------------------===//
#include "swift/IDE/KeyPathCompletion.h"
#include "swift/IDE/CodeCompletion.h"
#include "swift/IDE/CompletionLookup.h"
#include "swift/Sema/ConstraintSystem.h"
using namespace swift;
using namespace swift::constraints;
using namespace swift::ide;
void KeyPathTypeCheckCompletionCallback::sawSolutionImpl(
const constraints::Solution &S) {
// Determine the code completion.
size_t ComponentIndex = 0;
for (auto &Component : KeyPath->getComponents()) {
if (Component.getKind() == KeyPathExpr::Component::Kind::CodeCompletion) {
break;
} else {
ComponentIndex++;
}
}
assert(ComponentIndex < KeyPath->getComponents().size() &&
"Didn't find a code compleiton component?");
Type BaseType;
if (ComponentIndex == 0) {
// We are completing on the root and need to extract the key path's root
// type.
if (KeyPath->getRootType()) {
BaseType = S.getResolvedType(KeyPath->getRootType());
} else {
// The key path doesn't have a root TypeRepr set, so we can't look the key
// path's root up through it. Build a constraint locator and look the
// root type up through it.
// FIXME: Improve the linear search over S.typeBindings when it's possible
// to look up type variables by their locators.
auto RootLocator =
S.getConstraintLocator(KeyPath, {ConstraintLocator::KeyPathRoot});
auto BaseVariableTypeBinding =
llvm::find_if(S.typeBindings, [&RootLocator](const auto &Entry) {
return Entry.first->getImpl().getLocator() == RootLocator;
});
if (BaseVariableTypeBinding != S.typeBindings.end()) {
BaseType = S.simplifyType(BaseVariableTypeBinding->getSecond());
}
}
} else {
// We are completing after a component. Get the previous component's result
// type.
BaseType = S.simplifyType(S.getType(KeyPath, ComponentIndex - 1));
}
if (BaseType.isNull()) {
return;
}
// If ExpectedTy is a duplicate of any other result, ignore this solution.
if (llvm::any_of(Results, [&](const Result &R) {
return R.BaseType->isEqual(BaseType);
})) {
return;
}
Results.push_back({BaseType, /*OnRoot=*/(ComponentIndex == 0)});
}
void KeyPathTypeCheckCompletionCallback::collectResults(
DeclContext *DC, SourceLoc DotLoc,
ide::CodeCompletionContext &CompletionCtx) {
ASTContext &Ctx = DC->getASTContext();
CompletionLookup Lookup(CompletionCtx.getResultSink(), Ctx, DC,
&CompletionCtx);
if (DotLoc.isValid()) {
Lookup.setHaveDot(DotLoc);
}
Lookup.shouldCheckForDuplicates(Results.size() > 1);
for (auto &Result : Results) {
Lookup.setIsSwiftKeyPathExpr(Result.OnRoot);
Lookup.getValueExprCompletions(Result.BaseType);
}
collectCompletionResults(CompletionCtx, Lookup, DC, ExpectedTypeContext(),
/*CanCurrDeclContextHandleAsync=*/false);
}