1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
use std::any::Any;
use objc::Message;
use objc::runtime::{BOOL, Class, NO};
use objc_id::{Id, ShareId};
use NSString;
pub trait INSObject : Any + Sized + Message {
fn class() -> &'static Class;
fn hash_code(&self) -> usize {
unsafe {
msg_send![self, hash]
}
}
fn is_equal<T>(&self, other: &T) -> bool where T: INSObject {
let result: BOOL = unsafe {
msg_send![self, isEqual:other]
};
result != NO
}
fn description(&self) -> ShareId<NSString> {
unsafe {
let result: *mut NSString = msg_send![self, description];
Id::from_ptr(result)
}
}
fn is_kind_of(&self, cls: &Class) -> bool {
let result: BOOL = unsafe {
msg_send![self, isKindOfClass:cls]
};
result != NO
}
fn new() -> Id<Self> {
let cls = Self::class();
unsafe {
let obj: *mut Self = msg_send![cls, alloc];
let obj: *mut Self = msg_send![obj, init];
Id::from_retained_ptr(obj)
}
}
}
object_struct!(NSObject);
#[cfg(test)]
mod tests {
use {INSString, NSString};
use super::{INSObject, NSObject};
#[test]
fn test_is_equal() {
let obj1 = NSObject::new();
assert!(obj1.is_equal(&*obj1));
let obj2 = NSObject::new();
assert!(!obj1.is_equal(&*obj2));
}
#[test]
fn test_hash_code() {
let obj = NSObject::new();
assert!(obj.hash_code() == obj.hash_code());
}
#[test]
fn test_description() {
let obj = NSObject::new();
let description = obj.description();
let expected = format!("<NSObject: {:p}>", &*obj);
assert!(description.as_str() == &*expected);
}
#[test]
fn test_is_kind_of() {
let obj = NSObject::new();
assert!(obj.is_kind_of(NSObject::class()));
assert!(!obj.is_kind_of(NSString::class()));
}
}