|
| 1 | +use objc2::rc::Id; |
| 2 | +use objc2::runtime::{NSObject, NSObjectProtocol}; |
| 3 | +use objc2::{declare_class, extern_methods, extern_protocol, mutability, ClassType, ProtocolType}; |
| 4 | + |
| 5 | +extern_protocol!( |
| 6 | + #[allow(clippy::missing_safety_doc)] |
| 7 | + unsafe trait Proto: NSObjectProtocol { |
| 8 | + #[method(myMethod:)] |
| 9 | + fn protocol_method(mtm: MainThreadMarker, arg: i32) -> i32; |
| 10 | + |
| 11 | + #[method_id(myMethodId:)] |
| 12 | + fn protocol_method_id(mtm: MainThreadMarker, arg: &Self) -> Id<Self>; |
| 13 | + } |
| 14 | + |
| 15 | + unsafe impl ProtocolType for dyn Proto { |
| 16 | + const NAME: &'static str = "MainThreadMarkerTestProtocol"; |
| 17 | + } |
| 18 | +); |
| 19 | + |
| 20 | +declare_class!( |
| 21 | + #[derive(PartialEq, Eq, Hash, Debug)] |
| 22 | + struct Cls; |
| 23 | + |
| 24 | + unsafe impl ClassType for Cls { |
| 25 | + type Super = NSObject; |
| 26 | + type Mutability = mutability::InteriorMutable; |
| 27 | + const NAME: &'static str = "MainThreadMarkerTest"; |
| 28 | + } |
| 29 | + |
| 30 | + unsafe impl Proto for Cls { |
| 31 | + #[method(myMethod:)] |
| 32 | + fn _my_mainthreadonly_method(arg: i32) -> i32 { |
| 33 | + arg + 1 |
| 34 | + } |
| 35 | + |
| 36 | + #[method_id(myMethodId:)] |
| 37 | + fn _my_mainthreadonly_method_id(arg: &Self) -> Id<Self> { |
| 38 | + unsafe { Id::retain(arg as *const Self as *mut Self).unwrap() } |
| 39 | + } |
| 40 | + } |
| 41 | +); |
| 42 | + |
| 43 | +unsafe impl NSObjectProtocol for Cls {} |
| 44 | + |
| 45 | +// The macro does a textual match; but when users actually use |
| 46 | +// `icrate::Foundation::MainThreadMarker` to ensure soundness, they will not |
| 47 | +// do this! |
| 48 | +#[derive(Clone, Copy)] |
| 49 | +struct MainThreadMarker(bool); |
| 50 | + |
| 51 | +extern_methods!( |
| 52 | + unsafe impl Cls { |
| 53 | + #[method_id(new)] |
| 54 | + fn new() -> Id<Self>; |
| 55 | + |
| 56 | + #[method(myMethod:)] |
| 57 | + fn method(mtm: MainThreadMarker, arg: i32, mtm2: MainThreadMarker) -> i32; |
| 58 | + |
| 59 | + #[method_id(myMethodId:)] |
| 60 | + fn method_id(mtm: MainThreadMarker, arg: &Self, mtm2: MainThreadMarker) -> Id<Self>; |
| 61 | + } |
| 62 | +); |
| 63 | + |
| 64 | +#[test] |
| 65 | +fn call() { |
| 66 | + let obj1 = Cls::new(); |
| 67 | + let mtm = MainThreadMarker(true); |
| 68 | + |
| 69 | + let res = Cls::method(mtm, 2, mtm); |
| 70 | + assert_eq!(res, 3); |
| 71 | + let res = Cls::protocol_method(mtm, 3); |
| 72 | + assert_eq!(res, 4); |
| 73 | + |
| 74 | + let obj2 = Cls::method_id(mtm, &obj1, mtm); |
| 75 | + assert_eq!(obj1, obj2); |
| 76 | + |
| 77 | + let obj2 = Cls::protocol_method_id(mtm, &obj1); |
| 78 | + assert_eq!(obj1, obj2); |
| 79 | +} |
0 commit comments