summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorIlya Biryukov <ibiryukov@google.com>2019-05-28 15:33:37 +0000
committerIlya Biryukov <ibiryukov@google.com>2019-05-28 15:33:37 +0000
commit8534675cefb427f33c4fa083d2751ef09acccaf5 (patch)
treea592558488f33d1679f650fcd6aa3d096c30a3d2
parent756565d47079044b31b88bbcb8f71518c9526bd8 (diff)
downloadbcm5719-llvm-8534675cefb427f33c4fa083d2751ef09acccaf5.tar.gz
bcm5719-llvm-8534675cefb427f33c4fa083d2751ef09acccaf5.zip
[clangd] Place cursor better after completing patterns
Summary: By producing the $0 marker in the snippets at the last placeholder. This produces nicer results in most cases, e.g. for namespace <#name#> { <#decls#> } we now produce ${0:decls} instead of ${2:decls} and the final cursor placement is more convenient. Reviewers: hokein Reviewed By: hokein Subscribers: MaskRay, jkorous, arphaman, kadircet, cfe-commits Tags: #clang Differential Revision: https://reviews.llvm.org/D62389 llvm-svn: 361841
-rw-r--r--clang-tools-extra/clangd/CodeComplete.cpp3
-rw-r--r--clang-tools-extra/clangd/CodeCompletionStrings.cpp27
-rw-r--r--clang-tools-extra/clangd/CodeCompletionStrings.h8
-rw-r--r--clang-tools-extra/clangd/unittests/CodeCompleteTests.cpp22
-rw-r--r--clang-tools-extra/clangd/unittests/CodeCompletionStringsTests.cpp25
5 files changed, 76 insertions, 9 deletions
diff --git a/clang-tools-extra/clangd/CodeComplete.cpp b/clang-tools-extra/clangd/CodeComplete.cpp
index 7f811c31de5..4328a647f4b 100644
--- a/clang-tools-extra/clangd/CodeComplete.cpp
+++ b/clang-tools-extra/clangd/CodeComplete.cpp
@@ -394,8 +394,9 @@ struct CodeCompletionBuilder {
Bundled.emplace_back();
BundledEntry &S = Bundled.back();
if (C.SemaResult) {
+ bool IsPattern = C.SemaResult->Kind == CodeCompletionResult::RK_Pattern;
getSignature(*SemaCCS, &S.Signature, &S.SnippetSuffix,
- &Completion.RequiredQualifier);
+ &Completion.RequiredQualifier, IsPattern);
S.ReturnType = getReturnType(*SemaCCS);
} else if (C.IndexResult) {
S.Signature = C.IndexResult->Signature;
diff --git a/clang-tools-extra/clangd/CodeCompletionStrings.cpp b/clang-tools-extra/clangd/CodeCompletionStrings.cpp
index 586be67e92c..bf3cabc2698 100644
--- a/clang-tools-extra/clangd/CodeCompletionStrings.cpp
+++ b/clang-tools-extra/clangd/CodeCompletionStrings.cpp
@@ -11,6 +11,8 @@
#include "clang/AST/DeclObjC.h"
#include "clang/AST/RawCommentList.h"
#include "clang/Basic/SourceManager.h"
+#include "clang/Sema/CodeCompleteConsumer.h"
+#include <limits>
#include <utility>
namespace clang {
@@ -73,8 +75,23 @@ std::string getDeclComment(const ASTContext &Ctx, const NamedDecl &Decl) {
}
void getSignature(const CodeCompletionString &CCS, std::string *Signature,
- std::string *Snippet, std::string *RequiredQualifiers) {
- unsigned ArgCount = 0;
+ std::string *Snippet, std::string *RequiredQualifiers,
+ bool CompletingPattern) {
+ // Placeholder with this index will be ${0:…} to mark final cursor position.
+ // Usually we do not add $0, so the cursor is placed at end of completed text.
+ unsigned CursorSnippetArg = std::numeric_limits<unsigned>::max();
+ if (CompletingPattern) {
+ // In patterns, it's best to place the cursor at the last placeholder, to
+ // handle cases like
+ // namespace ${1:name} {
+ // ${0:decls}
+ // }
+ CursorSnippetArg =
+ llvm::count_if(CCS, [](const CodeCompletionString::Chunk &C) {
+ return C.Kind == CodeCompletionString::CK_Placeholder;
+ });
+ }
+ unsigned SnippetArg = 0;
bool HadObjCArguments = false;
for (const auto &Chunk : CCS) {
// Informative qualifier chunks only clutter completion results, skip
@@ -124,8 +141,10 @@ void getSignature(const CodeCompletionString &CCS, std::string *Signature,
break;
case CodeCompletionString::CK_Placeholder:
*Signature += Chunk.Text;
- ++ArgCount;
- *Snippet += "${" + std::to_string(ArgCount) + ':';
+ ++SnippetArg;
+ *Snippet +=
+ "${" +
+ std::to_string(SnippetArg == CursorSnippetArg ? 0 : SnippetArg) + ':';
appendEscapeSnippet(Chunk.Text, Snippet);
*Snippet += '}';
break;
diff --git a/clang-tools-extra/clangd/CodeCompletionStrings.h b/clang-tools-extra/clangd/CodeCompletionStrings.h
index 153e0af1189..6733d0231df 100644
--- a/clang-tools-extra/clangd/CodeCompletionStrings.h
+++ b/clang-tools-extra/clangd/CodeCompletionStrings.h
@@ -38,12 +38,16 @@ std::string getDeclComment(const ASTContext &Ctx, const NamedDecl &D);
/// Formats the signature for an item, as a display string and snippet.
/// e.g. for const_reference std::vector<T>::at(size_type) const, this returns:
/// *Signature = "(size_type) const"
-/// *Snippet = "(${0:size_type})"
+/// *Snippet = "(${1:size_type})"
/// If set, RequiredQualifiers is the text that must be typed before the name.
/// e.g "Base::" when calling a base class member function that's hidden.
+///
+/// When \p CompletingPattern is true, the last placeholder will be of the form
+/// ${0:…}, indicating the cursor should stay there.
void getSignature(const CodeCompletionString &CCS, std::string *Signature,
std::string *Snippet,
- std::string *RequiredQualifiers = nullptr);
+ std::string *RequiredQualifiers = nullptr,
+ bool CompletingPattern = false);
/// Assembles formatted documentation for a completion result. This includes
/// documentation comments and other relevant information like annotations.
diff --git a/clang-tools-extra/clangd/unittests/CodeCompleteTests.cpp b/clang-tools-extra/clangd/unittests/CodeCompleteTests.cpp
index 8f8376e25bc..4dcb8701a35 100644
--- a/clang-tools-extra/clangd/unittests/CodeCompleteTests.cpp
+++ b/clang-tools-extra/clangd/unittests/CodeCompleteTests.cpp
@@ -2382,6 +2382,28 @@ TEST(CompletionTest, ObjectiveCMethodTwoArgumentsFromMiddle) {
EXPECT_THAT(C, ElementsAre(SnippetSuffix("${1:(unsigned int)}")));
}
+TEST(CompletionTest, CursorInSnippets) {
+ clangd::CodeCompleteOptions Options;
+ Options.EnableSnippets = true;
+ auto Results = completions(
+ R"cpp(
+ void while_foo(int a, int b);
+ void test() {
+ whil^
+ })cpp",
+ /*IndexSymbols=*/{}, Options);
+
+ // Last placeholder in code patterns should be $0 to put the cursor there.
+ EXPECT_THAT(
+ Results.Completions,
+ Contains(AllOf(Named("while"),
+ SnippetSuffix("(${1:condition}){${0:statements}\n}"))));
+ // However, snippets for functions must *not* end with $0.
+ EXPECT_THAT(Results.Completions,
+ Contains(AllOf(Named("while_foo"),
+ SnippetSuffix("(${1:int a}, ${2:int b})"))));
+}
+
TEST(CompletionTest, WorksWithNullType) {
auto R = completions(R"cpp(
int main() {
diff --git a/clang-tools-extra/clangd/unittests/CodeCompletionStringsTests.cpp b/clang-tools-extra/clangd/unittests/CodeCompletionStringsTests.cpp
index 43429c86465..83b3826f6fc 100644
--- a/clang-tools-extra/clangd/unittests/CodeCompletionStringsTests.cpp
+++ b/clang-tools-extra/clangd/unittests/CodeCompletionStringsTests.cpp
@@ -22,10 +22,12 @@ public:
CCTUInfo(Allocator), Builder(*Allocator, CCTUInfo) {}
protected:
- void computeSignature(const CodeCompletionString &CCS) {
+ void computeSignature(const CodeCompletionString &CCS,
+ bool CompletingPattern = false) {
Signature.clear();
Snippet.clear();
- getSignature(CCS, &Signature, &Snippet);
+ getSignature(CCS, &Signature, &Snippet, /*RequiredQualifier=*/nullptr,
+ CompletingPattern);
}
std::shared_ptr<clang::GlobalCodeCompletionAllocator> Allocator;
@@ -99,6 +101,25 @@ TEST_F(CompletionStringTest, EscapeSnippet) {
EXPECT_EQ(Snippet, "(${1:\\$p\\}1\\\\})");
}
+TEST_F(CompletionStringTest, SnippetsInPatterns) {
+ auto MakeCCS = [this]() -> const CodeCompletionString & {
+ CodeCompletionBuilder Builder(*Allocator, CCTUInfo);
+ Builder.AddTypedTextChunk("namespace");
+ Builder.AddChunk(CodeCompletionString::CK_HorizontalSpace);
+ Builder.AddPlaceholderChunk("name");
+ Builder.AddChunk(CodeCompletionString::CK_Equal);
+ Builder.AddPlaceholderChunk("target");
+ Builder.AddChunk(CodeCompletionString::CK_SemiColon);
+ return *Builder.TakeString();
+ };
+ computeSignature(MakeCCS(), /*CompletingPattern=*/false);
+ EXPECT_EQ(Snippet, " ${1:name} = ${2:target};");
+
+ // When completing a pattern, the last placeholder holds the cursor position.
+ computeSignature(MakeCCS(), /*CompletingPattern=*/true);
+ EXPECT_EQ(Snippet, " ${1:name} = ${0:target};");
+}
+
TEST_F(CompletionStringTest, IgnoreInformativeQualifier) {
Builder.AddTypedTextChunk("X");
Builder.AddInformativeChunk("info ok");
OpenPOWER on IntegriCloud