diff --git a/clang-tools-extra/clangd/InlayHints.cpp b/clang-tools-extra/clangd/InlayHints.cpp --- a/clang-tools-extra/clangd/InlayHints.cpp +++ b/clang-tools-extra/clangd/InlayHints.cpp @@ -54,6 +54,8 @@ // FIXME: Handle RecoveryExpr to try to hint some invalid calls. private: + using NameVec = SmallVector; + // The purpose of Anchor is to deal with macros. It should be the call's // opening or closing parenthesis or brace. (Always using the opening would // make more sense but CallExpr only exposes the closing.) We heuristically @@ -74,16 +76,17 @@ if (Ctor->isCopyOrMoveConstructor()) return; - // FIXME: Exclude setters (i.e. functions with one argument whose name - // begins with "set"), as their parameter name is also not likely to be - // interesting. - // Don't show hints for variadic parameters. size_t FixedParamCount = getFixedParamCount(Callee); size_t ArgCount = std::min(FixedParamCount, Args.size()); NameVec ParameterNames = chooseParameterNames(Callee, ArgCount); + // Exclude setters (i.e. functions with one argument whose name begins with + // "set"), as their parameter name is also not likely to be interesting. + if (isSetter(Callee, ParameterNames)) + return; + for (size_t I = 0; I < ArgCount; ++I) { StringRef Name = ParameterNames[I]; if (!shouldHint(Args[I], Name)) @@ -94,6 +97,23 @@ } } + static bool isSetter(const FunctionDecl *Callee, const NameVec &ParamNames) { + if (ParamNames.size() != 1) + return false; + + StringRef Name = getSimpleName(*Callee); + if (!Name.startswith_lower("set")) + return false; + + // In addition to checking that the function has one parameter and its + // name starts with "set", also check that the part after "set" matches + // the name of the parameter (ignoring case). The idea here is that if + // the parameter name differs, it may contain extra information that + // may be useful to show in a hint, as in: + // void setTimeout(int timeoutMillis); + return Name.substr(3).equals_lower(ParamNames[0]); + } + bool shouldHint(const Expr *Arg, StringRef ParamName) { if (ParamName.empty()) return false; @@ -147,8 +167,6 @@ return {}; } - using NameVec = SmallVector; - NameVec chooseParameterNames(const FunctionDecl *Callee, size_t ArgCount) { // The current strategy here is to use all the parameter names from the // canonical declaration, unless they're all empty, in which case we diff --git a/clang-tools-extra/clangd/unittests/InlayHintTests.cpp b/clang-tools-extra/clangd/unittests/InlayHintTests.cpp --- a/clang-tools-extra/clangd/unittests/InlayHintTests.cpp +++ b/clang-tools-extra/clangd/unittests/InlayHintTests.cpp @@ -336,6 +336,23 @@ ExpectedHint{"param: ", "param"}); } +TEST(ParameterHints, SetterFunctions) { + assertParameterHints(R"cpp( + struct S { + void setParent(S* parent); + void setTimeout(int timeoutMillis); + }; + void bar() { + S s; + // Parameter name matches setter name - omit hint. + s.setParent(nullptr); + // Parameter name may contain extra info - show hint. + s.setTimeout($timeoutMillis[[120]]); + } + )cpp", + ExpectedHint{"timeoutMillis: ", "timeoutMillis"}); +} + } // namespace } // namespace clangd } // namespace clang