mirror of
https://github.com/apple/swift.git
synced 2025-12-14 20:36:38 +01:00
274 lines
8.2 KiB
Swift
274 lines
8.2 KiB
Swift
// RUN: %target-typecheck-verify-swift -enable-objc-interop
|
|
|
|
// Test requirements and conformance for Objective-C protocols.
|
|
|
|
@objc class ObjCClass { }
|
|
|
|
@objc protocol P1 {
|
|
func method1()
|
|
|
|
var property1: ObjCClass { get }
|
|
var property2: ObjCClass { get set }
|
|
}
|
|
|
|
@objc class C1 : P1 {
|
|
@objc(method1renamed)
|
|
func method1() { // expected-error{{Objective-C method 'method1renamed' provided by method 'method1()' does not match the requirement's selector ('method1')}}{{-1:9-23=method1}}
|
|
}
|
|
|
|
var property1: ObjCClass {
|
|
@objc(getProperty1) get { // expected-error{{Objective-C method 'getProperty1' provided by getter for 'property1' does not match the requirement's selector ('property1')}}{{11-23=property1}}
|
|
return ObjCClass()
|
|
}
|
|
}
|
|
|
|
var property2: ObjCClass {
|
|
get { return ObjCClass() }
|
|
@objc(setProperty2Please:) set { } // expected-error{{Objective-C method 'setProperty2Please:' provided by setter for 'property2' does not match the requirement's selector ('setProperty2:')}}{{11-30=setProperty2:}}
|
|
}
|
|
}
|
|
|
|
class C1b : P1 {
|
|
func method1() { }
|
|
var property1: ObjCClass = ObjCClass()
|
|
var property2: ObjCClass = ObjCClass()
|
|
}
|
|
|
|
@objc protocol P2 {
|
|
@objc(methodWithInt:withClass:)
|
|
func method(_: Int, class: ObjCClass)
|
|
|
|
var empty: Bool { @objc(checkIfEmpty) get }
|
|
}
|
|
|
|
class C2a : P2 {
|
|
func method(_: Int, class: ObjCClass) { }
|
|
|
|
var empty: Bool {
|
|
get { }
|
|
}
|
|
}
|
|
|
|
class C2b : P2 {
|
|
@objc func method(_: Int, class: ObjCClass) { }
|
|
|
|
@objc var empty: Bool {
|
|
@objc get { }
|
|
}
|
|
}
|
|
|
|
@objc protocol P3 {
|
|
@objc optional func doSomething(x: Int)
|
|
@objc optional func doSomething(y: Int)
|
|
}
|
|
|
|
class C3a : P3 {
|
|
@objc func doSomething(x: Int) { }
|
|
}
|
|
|
|
// Complain about optional requirements that aren't satisfied
|
|
// according to Swift, but would conflict in Objective-C.
|
|
@objc protocol OptP1 {
|
|
@objc optional func method() // expected-note 2{{requirement 'method()' declared here}}
|
|
|
|
@objc optional var property1: ObjCClass { get } // expected-note 2{{requirement 'property1' declared here}}
|
|
@objc optional var property2: ObjCClass { get set } // expected-note{{requirement 'property2' declared here}}
|
|
}
|
|
|
|
@objc class OptC1a : OptP1 { // expected-note 3{{class 'OptC1a' declares conformance to protocol 'OptP1' here}}
|
|
@objc(method) func otherMethod() { } // expected-error{{Objective-C method 'method' provided by method 'otherMethod()' conflicts with optional requirement method 'method()' in protocol 'OptP1'}}
|
|
// expected-note@-1{{rename method to match requirement 'method()'}}{{22-33=method}}
|
|
|
|
@objc var otherProp1: ObjCClass {
|
|
@objc(property1) get { return ObjCClass() } // expected-error{{Objective-C method 'property1' provided by getter for 'otherProp1' conflicts with optional requirement getter for 'property1' in protocol 'OptP1'}}
|
|
}
|
|
|
|
@objc var otherProp2: ObjCClass {
|
|
get { return ObjCClass() }
|
|
@objc(setProperty2:) set { } // expected-error{{Objective-C method 'setProperty2:' provided by setter for 'otherProp2' conflicts with optional requirement setter for 'property2' in protocol 'OptP1'}}
|
|
}
|
|
}
|
|
|
|
@objc class OptC1b : OptP1 { // expected-note 2{{class 'OptC1b' declares conformance to protocol 'OptP1' here}}
|
|
@objc(property1) func someMethod() { } // expected-error{{Objective-C method 'property1' provided by method 'someMethod()' conflicts with optional requirement getter for 'property1' in protocol 'OptP1'}}
|
|
|
|
@objc var someProp: ObjCClass {
|
|
@objc(method) get { return ObjCClass() } // expected-error{{Objective-C method 'method' provided by getter for 'someProp' conflicts with optional requirement method 'method()' in protocol 'OptP1'}}
|
|
}
|
|
}
|
|
|
|
// rdar://problem/19879598
|
|
@objc protocol Foo {
|
|
init()
|
|
}
|
|
|
|
class Bar: Foo {
|
|
required init() {}
|
|
}
|
|
|
|
@objc protocol P4 {
|
|
@objc(foo:bar:)
|
|
func method(x: Int, y: Int)
|
|
}
|
|
|
|
// Infer @objc and selector from requirement.
|
|
class C4a : P4 {
|
|
func method(x: Int, y: Int) { }
|
|
}
|
|
|
|
// Infer selector from requirement.
|
|
class C4b : P4 {
|
|
@objc
|
|
func method(x: Int, y: Int) { }
|
|
}
|
|
|
|
@objc protocol P5 {
|
|
@objc(wibble:wobble:)
|
|
func method(x: Int, y: Int)
|
|
}
|
|
|
|
// Don't infer when there is an ambiguity.
|
|
class C4_5a : P4, P5 {
|
|
func method(x: Int, y: Int) { }
|
|
// expected-error@-1{{ambiguous inference of Objective-C name for instance method 'method(x:y:)' ('foo:bar:' vs 'wibble:wobble:')}}
|
|
// expected-note@-2{{'method(x:y:)' (in protocol 'P4') provides Objective-C name 'foo:bar:'}}{{3-3=@objc(foo:bar:) }}
|
|
// expected-note@-3{{'method(x:y:)' (in protocol 'P5') provides Objective-C name 'wibble:wobble:'}}{{3-3=@objc(wibble:wobble:) }}
|
|
// expected-note@-4{{add '@nonobjc' to silence this error}}{{3-3=@nonobjc }}
|
|
// expected-error@-5{{Objective-C method 'foo:bar:' provided by method 'method(x:y:)' does not match the requirement's selector ('wibble:wobble:')}}
|
|
}
|
|
|
|
// Don't complain about a selector mismatch in cases where the
|
|
// selector will be inferred.
|
|
@objc protocol P6 {
|
|
func doSomething(_ sender: AnyObject?) // expected-note{{requirement 'doSomething' declared here}}
|
|
}
|
|
|
|
class C6a : P6 {
|
|
func doSomething(sender: AnyObject?) { } // expected-error{{method 'doSomething(sender:)' has different argument labels from those required by protocol 'P6' ('doSomething')}}{{20-20=_ }}{{none}}
|
|
}
|
|
|
|
|
|
|
|
@objc protocol P7 {
|
|
var prop: Int {
|
|
@objc(getTheProp) get
|
|
@objc(setTheProp:) set
|
|
}
|
|
}
|
|
|
|
class C7 : P7 {
|
|
var prop: Int {
|
|
get { return 0 }
|
|
set {}
|
|
}
|
|
}
|
|
|
|
class C7a : P7 {
|
|
@objc var prop: Int {
|
|
get { return 0 }
|
|
set {}
|
|
}
|
|
}
|
|
|
|
class C7b : P7 {
|
|
@objc var prop: Int {
|
|
@objc(getTheProp) get { return 0 }
|
|
@objc(setTheProp:) set {}
|
|
}
|
|
}
|
|
|
|
class C7c : P7 {
|
|
var prop: Int = 0
|
|
}
|
|
|
|
class C7d : P7 {
|
|
@objc var prop: Int = 0
|
|
}
|
|
|
|
class C7e : P7 {
|
|
// FIXME: This should probably still complain.
|
|
@objc(notProp) var prop: Int {
|
|
get { return 0 }
|
|
set {}
|
|
}
|
|
}
|
|
|
|
class C7f : P7 {
|
|
var prop: Int {
|
|
@objc(getProp) get { return 0 } // expected-error {{Objective-C method 'getProp' provided by getter for 'prop' does not match the requirement's selector ('getTheProp')}} {{11-18=getTheProp}}
|
|
set {}
|
|
}
|
|
}
|
|
|
|
class C7g : P7 {
|
|
var prop: Int {
|
|
get { return 0 }
|
|
@objc(prop:) set {} // expected-error {{Objective-C method 'prop:' provided by setter for 'prop' does not match the requirement's selector ('setTheProp:')}} {{11-16=setTheProp:}}
|
|
}
|
|
}
|
|
|
|
class C7h : P7 {
|
|
@objc var prop: Int = 0 {
|
|
didSet {}
|
|
}
|
|
}
|
|
|
|
class C7i : P7 {
|
|
@objc var prop: Int {
|
|
unsafeAddress { fatalError() }
|
|
unsafeMutableAddress { fatalError() }
|
|
}
|
|
}
|
|
|
|
@objc protocol P8 {
|
|
@objc optional var prop: Int {
|
|
@objc(getTheProp) get
|
|
}
|
|
}
|
|
|
|
class C8Base: P8 {}
|
|
class C8Sub: C8Base {
|
|
var prop: Int { return 0 } // expected-note {{getter for 'prop' declared here}}
|
|
|
|
@objc(getTheProp) func collision() {} // expected-error {{method 'collision()' with Objective-C selector 'getTheProp' conflicts with getter for 'prop' with the same Objective-C selector}}
|
|
}
|
|
class C8SubA: C8Base {
|
|
var prop: Int {
|
|
@objc get { return 0 } // expected-note {{getter for 'prop' declared here}}
|
|
}
|
|
|
|
@objc(getTheProp) func collision() {} // expected-error {{method 'collision()' with Objective-C selector 'getTheProp' conflicts with getter for 'prop' with the same Objective-C selector}}
|
|
}
|
|
class C8SubB: C8Base {
|
|
var prop: Int {
|
|
@objc(getProp) get { return 0 }
|
|
}
|
|
|
|
@objc(getTheProp) func collision() {} // okay
|
|
}
|
|
|
|
// These used to crash because the requirement is get-only.
|
|
class C8SubRW: C8Base {
|
|
var prop: Int {
|
|
get { return 0 }
|
|
set {}
|
|
}
|
|
}
|
|
|
|
class C8SubRW2: C8Base {
|
|
var prop: Int = 0
|
|
}
|
|
|
|
@available(macOS 12, iOS 15, tvOS 15, watchOS 8, *)
|
|
@objc protocol P9 {
|
|
@objc(custom:) func f(_: Any) // expected-warning {{method 'f' with Objective-C selector 'custom:' conflicts with method 'h()' with the same Objective-C selector; this is an error in the Swift 6 language mode}}
|
|
@objc(custom:) func g(_: Any) // expected-warning {{method 'g' with Objective-C selector 'custom:' conflicts with method 'h()' with the same Objective-C selector; this is an error in the Swift 6 language mode}}
|
|
@objc(custom:) func h() async // expected-note 2 {{method 'h()' declared here}}
|
|
}
|
|
|
|
// We forgot to emit diagnostics in the @objc path of StructuralRequirementsRequest.
|
|
struct MyStruct {}
|
|
|
|
@objc protocol InvalidProtocol where Self: MyStruct {}
|
|
// expected-error@-1 {{type 'Self' constrained to non-protocol, non-class type 'MyStruct'}}
|