|  | //===-- ClangMemberTests.cpp - unit tests for renaming class members ------===// | 
|  | // | 
|  | // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. | 
|  | // See https://llvm.org/LICENSE.txt for license information. | 
|  | // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception | 
|  | // | 
|  | //===----------------------------------------------------------------------===// | 
|  |  | 
|  | #include "ClangRenameTest.h" | 
|  |  | 
|  | namespace clang { | 
|  | namespace clang_rename { | 
|  | namespace test { | 
|  | namespace { | 
|  |  | 
|  | class RenameMemberTest : public ClangRenameTest { | 
|  | public: | 
|  | RenameMemberTest() { | 
|  | AppendToHeader(R"( | 
|  | struct NA { | 
|  | void Foo(); | 
|  | void NotFoo(); | 
|  | static void SFoo(); | 
|  | static void SNotFoo(); | 
|  | int Moo; | 
|  | }; | 
|  | struct A { | 
|  | virtual void Foo(); | 
|  | void NotFoo(); | 
|  | static void SFoo(); | 
|  | static void SNotFoo(); | 
|  | int Moo; | 
|  | int NotMoo; | 
|  | static int SMoo; | 
|  | }; | 
|  | struct B : public A { | 
|  | void Foo() override; | 
|  | }; | 
|  | template <typename T> struct TA { | 
|  | T* Foo(); | 
|  | T* NotFoo(); | 
|  | static T* SFoo(); | 
|  | static T* NotSFoo(); | 
|  | }; | 
|  | template <typename T> struct TB : public TA<T> {}; | 
|  | namespace ns { | 
|  | template <typename T> struct TA { | 
|  | T* Foo(); | 
|  | T* NotFoo(); | 
|  | static T* SFoo(); | 
|  | static T* NotSFoo(); | 
|  | static int SMoo; | 
|  | }; | 
|  | template <typename T> struct TB : public TA<T> {}; | 
|  | struct A { | 
|  | void Foo(); | 
|  | void NotFoo(); | 
|  | static void SFoo(); | 
|  | static void SNotFoo(); | 
|  | }; | 
|  | struct B : public A {}; | 
|  | struct C { | 
|  | template <class T> | 
|  | void SFoo(const T& t) {} | 
|  | template <class T> | 
|  | void Foo() {} | 
|  | }; | 
|  | })"); | 
|  | } | 
|  | }; | 
|  |  | 
|  | INSTANTIATE_TEST_CASE_P( | 
|  | DISABLED_RenameTemplatedClassStaticVariableTest, RenameMemberTest, | 
|  | testing::ValuesIn(std::vector<Case>({ | 
|  | // FIXME: support renaming static variables for template classes. | 
|  | {"void f() { ns::TA<int>::SMoo; }", | 
|  | "void f() { ns::TA<int>::SMeh; }", "ns::TA::SMoo", "ns::TA::SMeh"}, | 
|  | })), ); | 
|  |  | 
|  | INSTANTIATE_TEST_CASE_P( | 
|  | RenameMemberTest, RenameMemberTest, | 
|  | testing::ValuesIn(std::vector<Case>({ | 
|  | // Normal methods and fields. | 
|  | {"void f() { A a; a.Foo(); }", "void f() { A a; a.Bar(); }", "A::Foo", | 
|  | "A::Bar"}, | 
|  | {"void f() { ns::A a; a.Foo(); }", "void f() { ns::A a; a.Bar(); }", | 
|  | "ns::A::Foo", "ns::A::Bar"}, | 
|  | {"void f() { A a; int x = a.Moo; }", "void f() { A a; int x = a.Meh; }", | 
|  | "A::Moo", "A::Meh"}, | 
|  | {"void f() { B b; b.Foo(); }", "void f() { B b; b.Bar(); }", "B::Foo", | 
|  | "B::Bar"}, | 
|  | {"void f() { ns::B b; b.Foo(); }", "void f() { ns::B b; b.Bar(); }", | 
|  | "ns::A::Foo", "ns::A::Bar"}, | 
|  | {"void f() { B b; int x = b.Moo; }", "void f() { B b; int x = b.Meh; }", | 
|  | "A::Moo", "A::Meh"}, | 
|  |  | 
|  | // Static methods. | 
|  | {"void f() { A::SFoo(); }", "void f() { A::SBar(); }", "A::SFoo", | 
|  | "A::SBar"}, | 
|  | {"void f() { ns::A::SFoo(); }", "void f() { ns::A::SBar(); }", | 
|  | "ns::A::SFoo", "ns::A::SBar"}, | 
|  | {"void f() { TA<int>::SFoo(); }", "void f() { TA<int>::SBar(); }", | 
|  | "TA::SFoo", "TA::SBar"}, | 
|  | {"void f() { ns::TA<int>::SFoo(); }", | 
|  | "void f() { ns::TA<int>::SBar(); }", "ns::TA::SFoo", "ns::TA::SBar"}, | 
|  |  | 
|  | // Static variables. | 
|  | {"void f() { A::SMoo; }", | 
|  | "void f() { A::SMeh; }", "A::SMoo", "A::SMeh"}, | 
|  |  | 
|  | // Templated methods. | 
|  | {"void f() { TA<int> a; a.Foo(); }", "void f() { TA<int> a; a.Bar(); }", | 
|  | "TA::Foo", "TA::Bar"}, | 
|  | {"void f() { ns::TA<int> a; a.Foo(); }", | 
|  | "void f() { ns::TA<int> a; a.Bar(); }", "ns::TA::Foo", "ns::TA::Bar"}, | 
|  | {"void f() { TB<int> b; b.Foo(); }", "void f() { TB<int> b; b.Bar(); }", | 
|  | "TA::Foo", "TA::Bar"}, | 
|  | {"void f() { ns::TB<int> b; b.Foo(); }", | 
|  | "void f() { ns::TB<int> b; b.Bar(); }", "ns::TA::Foo", "ns::TA::Bar"}, | 
|  | {"void f() { ns::C c; int x; c.SFoo(x); }", | 
|  | "void f() { ns::C c; int x; c.SBar(x); }", "ns::C::SFoo", | 
|  | "ns::C::SBar"}, | 
|  | {"void f() { ns::C c; c.Foo<int>(); }", | 
|  | "void f() { ns::C c; c.Bar<int>(); }", "ns::C::Foo", "ns::C::Bar"}, | 
|  |  | 
|  | // Pointers to methods. | 
|  | {"void f() { auto p = &A::Foo; }", "void f() { auto p = &A::Bar; }", | 
|  | "A::Foo", "A::Bar"}, | 
|  | {"void f() { auto p = &A::SFoo; }", "void f() { auto p = &A::SBar; }", | 
|  | "A::SFoo", "A::SBar"}, | 
|  | {"void f() { auto p = &B::Foo; }", "void f() { auto p = &B::Bar; }", | 
|  | "B::Foo", "B::Bar"}, | 
|  | {"void f() { auto p = &ns::A::Foo; }", | 
|  | "void f() { auto p = &ns::A::Bar; }", "ns::A::Foo", "ns::A::Bar"}, | 
|  | {"void f() { auto p = &ns::A::SFoo; }", | 
|  | "void f() { auto p = &ns::A::SBar; }", "ns::A::SFoo", "ns::A::SBar"}, | 
|  | {"void f() { auto p = &ns::C::SFoo<int>; }", | 
|  | "void f() { auto p = &ns::C::SBar<int>; }", "ns::C::SFoo", | 
|  | "ns::C::SBar"}, | 
|  |  | 
|  | // These methods are not declared or overridden in the subclass B, we | 
|  | // have to use the qualified name with parent class A to identify them. | 
|  | {"void f() { auto p = &ns::B::Foo; }", | 
|  | "void f() { auto p = &ns::B::Bar; }", "ns::A::Foo", "ns::B::Bar"}, | 
|  | {"void f() { B::SFoo(); }", "void f() { B::SBar(); }", "A::SFoo", | 
|  | "B::SBar"}, | 
|  | {"void f() { ns::B::SFoo(); }", "void f() { ns::B::SBar(); }", | 
|  | "ns::A::SFoo", "ns::B::SBar"}, | 
|  | {"void f() { auto p = &B::SFoo; }", "void f() { auto p = &B::SBar; }", | 
|  | "A::SFoo", "B::SBar"}, | 
|  | {"void f() { auto p = &ns::B::SFoo; }", | 
|  | "void f() { auto p = &ns::B::SBar; }", "ns::A::SFoo", "ns::B::SBar"}, | 
|  | {"void f() { TB<int>::SFoo(); }", "void f() { TB<int>::SBar(); }", | 
|  | "TA::SFoo", "TB::SBar"}, | 
|  | {"void f() { ns::TB<int>::SFoo(); }", | 
|  | "void f() { ns::TB<int>::SBar(); }", "ns::TA::SFoo", "ns::TB::SBar"}, | 
|  | })), ); | 
|  |  | 
|  | TEST_P(RenameMemberTest, RenameMembers) { | 
|  | auto Param = GetParam(); | 
|  | assert(!Param.OldName.empty()); | 
|  | assert(!Param.NewName.empty()); | 
|  | std::string Actual = | 
|  | runClangRenameOnCode(Param.Before, Param.OldName, Param.NewName); | 
|  | CompareSnippets(Param.After, Actual); | 
|  | } | 
|  |  | 
|  | TEST_F(RenameMemberTest, RenameMemberInsideClassMethods) { | 
|  | std::string Before = R"( | 
|  | struct X { | 
|  | int Moo; | 
|  | void Baz() { Moo = 1; } | 
|  | };)"; | 
|  | std::string Expected = R"( | 
|  | struct X { | 
|  | int Meh; | 
|  | void Baz() { Meh = 1; } | 
|  | };)"; | 
|  | std::string After = runClangRenameOnCode(Before, "X::Moo", "Y::Meh"); | 
|  | CompareSnippets(Expected, After); | 
|  | } | 
|  |  | 
|  | TEST_F(RenameMemberTest, RenameMethodInsideClassMethods) { | 
|  | std::string Before = R"( | 
|  | struct X { | 
|  | void Foo() {} | 
|  | void Baz() { Foo(); } | 
|  | };)"; | 
|  | std::string Expected = R"( | 
|  | struct X { | 
|  | void Bar() {} | 
|  | void Baz() { Bar(); } | 
|  | };)"; | 
|  | std::string After = runClangRenameOnCode(Before, "X::Foo", "X::Bar"); | 
|  | CompareSnippets(Expected, After); | 
|  | } | 
|  |  | 
|  | TEST_F(RenameMemberTest, RenameCtorInitializer) { | 
|  | std::string Before = R"( | 
|  | class X { | 
|  | public: | 
|  | X(); | 
|  | A a; | 
|  | A a2; | 
|  | B b; | 
|  | }; | 
|  |  | 
|  | X::X():a(), b() {} | 
|  | )"; | 
|  | std::string Expected = R"( | 
|  | class X { | 
|  | public: | 
|  | X(); | 
|  | A bar; | 
|  | A a2; | 
|  | B b; | 
|  | }; | 
|  |  | 
|  | X::X():bar(), b() {} | 
|  | )"; | 
|  | std::string After = runClangRenameOnCode(Before, "X::a", "X::bar"); | 
|  | CompareSnippets(Expected, After); | 
|  | } | 
|  |  | 
|  | } // anonymous namespace | 
|  | } // namespace test | 
|  | } // namespace clang_rename | 
|  | } // namesdpace clang |