Macro objc2::extern_methods
source · macro_rules! extern_methods { ( $( $(#[$impl_m:meta])* unsafe impl<$($t:ident $(: $b:ident $(+ $rest:ident)*)?),* $(,)?> $type:ty { $($methods:tt)* } )+ ) => { ... }; ( $( $(#[$impl_m:meta])* unsafe impl $type:ty { $($methods:tt)* } )+ ) => { ... }; }
Expand description
Define methods on an external class.
This is a convenience macro to generate associated functions and methods
that call msg_send!
or msg_send_id!
appropriately.
§Specification
Within the impl
block you can define two types of functions without
bodies; “associated functions” and “methods”. These are then mapped to
the Objective-C equivalents “class methods” and “instance methods”, and an
appropriate body is created for you. In particular, if you use self
or
the special name this
(or _this
), your method will assumed to be an
instance method, and if you don’t it will be assumed to be a class method.
The desired selector can be specified using the #[method(my:selector:)]
or #[method_id(my:selector:)]
attribute. The method
attribute maps to
a call to msg_send!
, while the method_id
maps to msg_send_id!
.
If the attribute ends with “_”, as in #[method(my:error:_)]
or
#[method_id(my:error:_)]
, the method is assumed to take an
implicit NSError**
parameter, which is automatically converted to a
Result
. See the error section in msg_send!
and msg_send_id!
for details.
If you use objc2_foundation::MainThreadMarker
as a parameter type, the
macro will ignore it, allowing you to neatly specify “this method must be
run on the main thread”. Note that due to type-system limitations, this is
currently a textual match on MainThreadMarker
; so you must use that
exact identifier.
Putting other attributes on the method such as cfg
, allow
, doc
,
deprecated
and so on is supported. However, note that cfg_attr
may not
work correctly, due to implementation difficulty - if you have a concrete
use-case, please open an issue, then we can discuss it.
The name of the function will be used for the resulting function that the user will use to access the functionality, but is otherwise not used by the macro.
If you specify a function/method with a body, the macro will output it unchanged.
§Safety
You must ensure that any methods you declare with the #[method(...)]
attribute upholds the safety guarantees decribed in the msg_send!
macro, or are marked unsafe
.
Likewise, you must ensure that any methods you declare with the
#[method_id(...)]
attribute upholds the safety guarantees decribed in
the msg_send_id!
macro, or are marked unsafe
.
§Examples
Let’s create a quick custom class:
use objc2::encode::{Encode, Encoding};
use objc2::ffi::NSUInteger;
use objc2::rc::{Allocated, Id};
use objc2::runtime::NSObject;
use objc2::{declare_class, extern_methods, mutability, ClassType, DeclaredClass};
// Shim
type NSError = NSObject;
declare_class!(
pub struct MyObject;
// SAFETY:
// - The superclass NSObject does not have any subclassing requirements.
// - Interior mutability is a safe default.
// - `MyObject` does not implement `Drop`.
unsafe impl ClassType for MyObject {
type Super = NSObject;
type Mutability = mutability::InteriorMutable;
const NAME: &'static str = "MyObject";
}
impl DeclaredClass for MyObject {}
unsafe impl MyObject {
// ... Assume we've implemented all the methods used below
}
);
extern_methods!(
/// Creation methods.
unsafe impl MyObject {
#[method_id(new)]
pub fn new() -> Id<Self>;
#[method_id(initWithVal:)]
// Arbitary self types are not stable, but we can work around it
// with the special name `this`.
pub fn init(this: Allocated<Self>, val: usize) -> Id<Self>;
}
/// Instance accessor methods.
unsafe impl MyObject {
#[method(foo)]
pub fn foo(&self) -> NSUInteger;
#[method_id(fooObject)]
pub fn foo_object(&self) -> Id<NSObject>;
#[method(withError:_)]
// Since the selector specifies "_", the return type is assumed to
// be `Result`.
pub fn with_error(&self) -> Result<(), Id<NSError>>;
}
);
The extern_methods!
declaration then becomes:
use objc2::{msg_send, msg_send_id};
/// Creation methods.
impl MyObject {
pub fn new() -> Id<Self> {
unsafe { msg_send_id![Self::class(), new] }
}
pub fn init(this: Allocated<Self>, val: usize) -> Id<Self> {
unsafe { msg_send_id![this, initWithVal: val] }
}
}
/// Instance accessor methods.
impl MyObject {
pub fn foo(&self) -> NSUInteger {
unsafe { msg_send![self, foo] }
}
pub fn foo_object(&self) -> Id<NSObject> {
unsafe { msg_send_id![self, fooObject] }
}
// Since the selector specifies one more argument than we
// have, the return type is assumed to be `Result`.
pub fn with_error(&self) -> Result<(), Id<NSError>> {
unsafe { msg_send![self, withError: _] }
}
}
See the source code of objc2-foundation
for many more examples.