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
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
use core::ffi::c_void;
use core::fmt;
use core::ptr::NonNull;
use core::slice;
use core::str;
use std::os::raw::c_char;
use objc2::msg_send;
use objc2::rc::{Id, Owned, Shared};
use super::INSObject;
pub trait INSCopying: INSObject {
type Output: INSObject;
fn copy(&self) -> Id<Self::Output, Shared> {
unsafe {
let obj: *mut Self::Output = msg_send![self, copy];
Id::new(NonNull::new_unchecked(obj))
}
}
}
pub trait INSMutableCopying: INSObject {
type Output: INSObject;
fn mutable_copy(&self) -> Id<Self::Output, Owned> {
unsafe {
let obj: *mut Self::Output = msg_send![self, mutableCopy];
Id::new(NonNull::new_unchecked(obj))
}
}
}
#[cfg(target_vendor = "apple")]
const UTF8_ENCODING: usize = 4;
#[cfg(not(target_vendor = "apple"))]
const UTF8_ENCODING: i32 = 4;
pub trait INSString: INSObject {
fn len(&self) -> usize {
unsafe { msg_send![self, lengthOfBytesUsingEncoding: UTF8_ENCODING] }
}
fn is_empty(&self) -> bool {
self.len() == 0
}
fn as_str(&self) -> &str {
let bytes = unsafe {
let bytes: *const c_char = msg_send![self, UTF8String];
bytes as *const u8
};
let len = self.len();
unsafe {
let bytes = slice::from_raw_parts(bytes, len);
str::from_utf8(bytes).unwrap()
}
}
fn from_str(string: &str) -> Id<Self, Owned> {
let cls = Self::class();
let bytes = string.as_ptr() as *const c_void;
unsafe {
let obj: *mut Self = msg_send![cls, alloc];
let obj: *mut Self = msg_send![
obj,
initWithBytes: bytes,
length: string.len(),
encoding: UTF8_ENCODING,
];
Id::new(NonNull::new_unchecked(obj))
}
}
}
object_struct!(NSString);
impl INSString for NSString {}
impl INSCopying for NSString {
type Output = NSString;
}
impl fmt::Display for NSString {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
fmt::Display::fmt(self.as_str(), f)
}
}
#[cfg(test)]
mod tests {
use super::{INSCopying, INSString, NSString};
#[cfg(not(target_vendor = "apple"))]
#[test]
fn ensure_linkage() {
unsafe { objc2::__gnustep_hack::get_class_to_force_linkage() };
}
#[test]
fn test_utf8() {
let expected = "ประเทศไทย中华Việt Nam";
let s = NSString::from_str(expected);
assert!(s.len() == expected.len());
assert!(s.as_str() == expected);
}
#[test]
fn test_interior_nul() {
let expected = "Hello\0World";
let s = NSString::from_str(expected);
assert!(s.len() == expected.len());
assert!(s.as_str() == expected);
}
#[test]
fn test_copy() {
let s = NSString::from_str("Hello!");
let copied = s.copy();
assert!(copied.as_str() == s.as_str());
}
}