mirror of
https://github.com/intel/llvm.git
synced 2026-02-08 00:50:03 +08:00
When completing Objective-C instance method invocations, perform a contextual conversion to an Objective-C pointer type of the target expression if needed. This fixes code completion of method invocations where the target is a smart pointer that has an explicit conversion operator to an Objective-C type.
llvm-svn: 202529
This commit is contained in:
@@ -5529,7 +5529,7 @@ void Sema::CodeCompleteObjCInstanceMessage(Scope *S, Expr *Receiver,
|
||||
// If we're messaging an expression with type "id" or "Class", check
|
||||
// whether we know something special about the receiver that allows
|
||||
// us to assume a more-specific receiver type.
|
||||
if (ReceiverType->isObjCIdType() || ReceiverType->isObjCClassType())
|
||||
if (ReceiverType->isObjCIdType() || ReceiverType->isObjCClassType()) {
|
||||
if (ObjCInterfaceDecl *IFace = GetAssumedMessageSendExprType(RecExpr)) {
|
||||
if (ReceiverType->isObjCClassType())
|
||||
return CodeCompleteObjCClassMessage(S,
|
||||
@@ -5540,6 +5540,13 @@ void Sema::CodeCompleteObjCInstanceMessage(Scope *S, Expr *Receiver,
|
||||
ReceiverType = Context.getObjCObjectPointerType(
|
||||
Context.getObjCInterfaceType(IFace));
|
||||
}
|
||||
} else if (RecExpr && getLangOpts().CPlusPlus) {
|
||||
ExprResult Conv = PerformContextuallyConvertToObjCPointer(RecExpr);
|
||||
if (Conv.isUsable()) {
|
||||
RecExpr = Conv.take();
|
||||
ReceiverType = RecExpr->getType();
|
||||
}
|
||||
}
|
||||
|
||||
// Build the set of methods we can see.
|
||||
ResultBuilder Results(*this, CodeCompleter->getAllocator(),
|
||||
|
||||
46
clang/test/CodeCompletion/objc-message.mm
Normal file
46
clang/test/CodeCompletion/objc-message.mm
Normal file
@@ -0,0 +1,46 @@
|
||||
// Note: the run lines follow their respective tests, since line/column
|
||||
// matter in this test.
|
||||
|
||||
@protocol FooTestProtocol
|
||||
+ protocolClassMethod;
|
||||
- protocolInstanceMethod;
|
||||
@end
|
||||
@interface Foo <FooTestProtocol> {
|
||||
void *isa;
|
||||
}
|
||||
+ (int)classMethod1:a withKeyword:b;
|
||||
+ (void)classMethod2;
|
||||
+ new;
|
||||
- instanceMethod1;
|
||||
@end
|
||||
|
||||
@interface Foo (FooTestCategory)
|
||||
+ categoryClassMethod;
|
||||
- categoryInstanceMethod;
|
||||
@end
|
||||
|
||||
template<typename T> struct RetainPtr {
|
||||
template <typename U> struct RemovePointer { typedef U Type; };
|
||||
template <typename U> struct RemovePointer<U*> { typedef U Type; };
|
||||
|
||||
typedef typename RemovePointer<T>::Type* PtrType;
|
||||
|
||||
explicit operator PtrType() const;
|
||||
};
|
||||
|
||||
void func(const RetainPtr<Foo>& ptr)
|
||||
{
|
||||
[ptr instanceMethod1];
|
||||
}
|
||||
|
||||
void func(const RetainPtr<id <FooTestProtocol>>& ptr)
|
||||
{
|
||||
[ptr instanceMethod1];
|
||||
}
|
||||
|
||||
// RUN: %clang_cc1 -fsyntax-only -std=c++11 -code-completion-at=%s:33:7 %s -o - | FileCheck -check-prefix=CHECK-CC1 %s
|
||||
// CHECK-CC1: categoryInstanceMethod : [#id#]categoryInstanceMethod
|
||||
// CHECK-CC1: instanceMethod1 : [#id#]instanceMethod1
|
||||
// CHECK-CC1: protocolInstanceMethod : [#id#]protocolInstanceMethod
|
||||
// RUN: %clang_cc1 -fsyntax-only -std=c++11 -code-completion-at=%s:38:7 %s -o - | FileCheck -check-prefix=CHECK-CC2 %s
|
||||
// CHECK-CC2: protocolInstanceMethod : [#id#]protocolInstanceMethod
|
||||
Reference in New Issue
Block a user