Skip to content

Commit

Permalink
WIP
Browse files Browse the repository at this point in the history
msg_send_id![super]
  • Loading branch information
madsmtm committed Oct 5, 2023
1 parent cedec25 commit 42dac6d
Show file tree
Hide file tree
Showing 72 changed files with 4,842 additions and 4,815 deletions.
46 changes: 22 additions & 24 deletions crates/icrate/examples/browser.rs
Original file line number Diff line number Diff line change
@@ -1,5 +1,5 @@
#![deny(unsafe_op_in_unsafe_fn)]
use core::{cell::OnceCell, ptr::NonNull};
use core::cell::OnceCell;

use icrate::{
AppKit::{
Expand All @@ -18,26 +18,23 @@ use icrate::{
WebKit::{WKNavigation, WKNavigationDelegate, WKWebView},
};
use objc2::{
declare::{Ivar, IvarDrop},
declare_class, msg_send, msg_send_id,
declare_class, msg_send_id,
mutability::MainThreadOnly,
rc::Id,
rc::{Allocated, Id},
runtime::{AnyObject, ProtocolObject, Sel},
sel, ClassType,
sel, ClassType, DeclaredClass,
};

type IdCell<T> = Box<OnceCell<Id<T>>>;

macro_rules! idcell {
($name:ident => $this:expr) => {
$this.$name.set($name).expect(&format!(
$this.ivars().$name.set($name).expect(&format!(
"ivar should not already be initialized: `{}`",
stringify!($name)
));
};
($name:ident <= $this:expr) => {
#[rustfmt::skip]
let Some($name) = $this.$name.get() else {
let Some($name) = $this.ivars().$name.get() else {
unreachable!(
"ivar should be initialized: `{}`",
stringify!($name)
Expand All @@ -46,30 +43,31 @@ macro_rules! idcell {
};
}

#[derive(Default)]
struct Ivars {
nav_url: OnceCell<Id<NSTextField>>,
web_view: OnceCell<Id<WKWebView>>,
window: OnceCell<Id<NSWindow>>,
}

declare_class!(
struct Delegate {
nav_url: IvarDrop<IdCell<NSTextField>, "_nav_url">,
web_view: IvarDrop<IdCell<WKWebView>, "_web_view">,
window: IvarDrop<IdCell<NSWindow>, "_window">,
}
mod ivars;
struct Delegate;

unsafe impl ClassType for Delegate {
type Super = NSObject;
type Mutability = MainThreadOnly;
const NAME: &'static str = "Delegate";
}

impl DeclaredClass for Delegate {
type Ivars = Ivars;
}

unsafe impl Delegate {
#[method(init)]
unsafe fn init(this: *mut Self) -> Option<NonNull<Self>> {
let this: Option<&mut Self> = msg_send![super(this), init];
this.map(|this| {
Ivar::write(&mut this.nav_url, IdCell::default());
Ivar::write(&mut this.web_view, IdCell::default());
Ivar::write(&mut this.window, IdCell::default());
NonNull::from(this)
})
#[method_id(init)]
fn init(this: Allocated<Self>) -> Option<Id<Self>> {
let this = this.set_ivars(Ivars::default());
unsafe { msg_send_id![super(this), init] }
}
}

Expand Down
60 changes: 30 additions & 30 deletions crates/icrate/examples/delegate.rs
Original file line number Diff line number Diff line change
@@ -1,52 +1,52 @@
#![deny(unsafe_op_in_unsafe_fn)]
use std::ptr::NonNull;

use icrate::AppKit::{NSApplication, NSApplicationActivationPolicyRegular, NSApplicationDelegate};
use icrate::Foundation::{
ns_string, MainThreadMarker, NSCopying, NSNotification, NSObject, NSObjectProtocol, NSString,
};
use objc2::declare::{Ivar, IvarBool, IvarDrop, IvarEncode};
use objc2::rc::Id;
use objc2::rc::{Allocated, Id};
use objc2::runtime::ProtocolObject;
use objc2::{declare_class, msg_send, msg_send_id, mutability, ClassType};
use objc2::{declare_class, msg_send_id, mutability, ClassType, DeclaredClass};

#[derive(Debug)]
#[allow(unused)]
struct Ivars {
ivar: u8,
another_ivar: bool,
box_ivar: Box<i32>,
maybe_box_ivar: Option<Box<i32>>,
id_ivar: Id<NSString>,
maybe_id_ivar: Option<Id<NSString>>,
}

declare_class!(
#[derive(Debug)]
struct AppDelegate {
ivar: IvarEncode<u8, "_ivar">,
another_ivar: IvarBool<"_another_ivar">,
box_ivar: IvarDrop<Box<i32>, "_box_ivar">,
maybe_box_ivar: IvarDrop<Option<Box<i32>>, "_maybe_box_ivar">,
id_ivar: IvarDrop<Id<NSString>, "_id_ivar">,
maybe_id_ivar: IvarDrop<Option<Id<NSString>>, "_maybe_id_ivar">,
}

mod ivars;
struct AppDelegate;

unsafe impl ClassType for AppDelegate {
type Super = NSObject;
type Mutability = mutability::MainThreadOnly;
const NAME: &'static str = "MyAppDelegate";
}

impl DeclaredClass for AppDelegate {
type Ivars = Ivars;
}

unsafe impl AppDelegate {
#[method(initWith:another:)]
unsafe fn init_with(
this: *mut Self,
#[method_id(initWith:another:)]
fn init_with(
this: Allocated<Self>,
ivar: u8,
another_ivar: bool,
) -> Option<NonNull<Self>> {
let this: Option<&mut Self> = unsafe { msg_send![super(this), init] };

this.map(|this| {
Ivar::write(&mut this.ivar, ivar);
Ivar::write(&mut this.another_ivar, another_ivar);
Ivar::write(&mut this.maybe_box_ivar, None);
Ivar::write(&mut this.maybe_id_ivar, Some(ns_string!("def").copy()));
Ivar::write(&mut this.box_ivar, Box::new(2));
Ivar::write(&mut this.id_ivar, NSString::from_str("abc"));
NonNull::from(this)
})
) -> Option<Id<Self>> {
let this = this.set_ivars(Ivars {
ivar,another_ivar,
box_ivar: Box::new(2),
maybe_box_ivar: None,
id_ivar: NSString::from_str("abc"),
maybe_id_ivar: Some(ns_string!("def").copy()),
});
unsafe { msg_send_id![super(this), init] }
}
}

Expand Down
56 changes: 28 additions & 28 deletions crates/icrate/examples/metal.rs
Original file line number Diff line number Diff line change
Expand Up @@ -20,12 +20,11 @@ use icrate::{
MetalKit::{MTKView, MTKViewDelegate},
};
use objc2::{
declare::{Ivar, IvarDrop},
declare_class, msg_send, msg_send_id,
declare_class, msg_send_id,
mutability::MainThreadOnly,
rc::Id,
rc::{Allocated, Id},
runtime::ProtocolObject,
ClassType,
ClassType, DeclaredClass,
};

#[rustfmt::skip]
Expand Down Expand Up @@ -101,18 +100,16 @@ pub struct Color {
pub b: f32,
}

type IdCell<T> = Box<OnceCell<Id<T>>>;

macro_rules! idcell {
($name:ident => $this:expr) => {
$this.$name.set($name).expect(&format!(
$this.ivars().$name.set($name).expect(&format!(
"ivar should not already be initialized: `{}`",
stringify!($name)
));
};
($name:ident <= $this:expr) => {
#[rustfmt::skip]
let Some($name) = $this.$name.get() else {
let Some($name) = $this.ivars().$name.get() else {
unreachable!(
"ivar should be initialized: `{}`",
stringify!($name)
Expand All @@ -121,17 +118,17 @@ macro_rules! idcell {
};
}

// declare the desired instance variables
struct Ivars {
start_date: Id<NSDate>,
command_queue: OnceCell<Id<ProtocolObject<dyn MTLCommandQueue>>>,
pipeline_state: OnceCell<Id<ProtocolObject<dyn MTLRenderPipelineState>>>,
window: OnceCell<Id<NSWindow>>,
}

// declare the Objective-C class machinery
declare_class!(
// declare the delegate class with our instance variables
#[rustfmt::skip] // FIXME: rustfmt breaks the macro parsing apparently
struct Delegate {
start_date: IvarDrop<Id<NSDate>, "_start_date">,
command_queue: IvarDrop<IdCell<ProtocolObject<dyn MTLCommandQueue>>, "_command_queue">,
pipeline_state: IvarDrop<IdCell<ProtocolObject<dyn MTLRenderPipelineState>>, "_pipeline_state">,
window: IvarDrop<IdCell<NSWindow>, "_window">,
}
mod ivars;
struct Delegate;

// declare the class type
unsafe impl ClassType for Delegate {
Expand All @@ -140,18 +137,21 @@ declare_class!(
const NAME: &'static str = "Delegate";
}

impl DeclaredClass for Delegate {
type Ivars = Ivars;
}

// define the Delegate methods (e.g., initializer)
unsafe impl Delegate {
#[method(init)]
unsafe fn init(this: *mut Self) -> Option<NonNull<Self>> {
let this: Option<&mut Self> = msg_send![super(this), init];
this.map(|this| {
Ivar::write(&mut this.start_date, unsafe { NSDate::now() });
Ivar::write(&mut this.command_queue, IdCell::default());
Ivar::write(&mut this.pipeline_state, IdCell::default());
Ivar::write(&mut this.window, IdCell::default());
NonNull::from(this)
})
#[method_id(init)]
fn init(this: Allocated<Self>) -> Option<Id<Self>> {
let this = this.set_ivars(Ivars {
start_date: unsafe { NSDate::now() },
command_queue: OnceCell::default(),
pipeline_state: OnceCell::default(),
window: OnceCell::default(),
});
unsafe { msg_send_id![super(this), init] }
}
}

Expand Down Expand Up @@ -277,7 +277,7 @@ declare_class!(

// compute the scene properties
let scene_properties_data = &SceneProperties {
time: unsafe { self.start_date.timeIntervalSinceNow() } as f32,
time: unsafe { self.ivars().start_date.timeIntervalSinceNow() } as f32,
};
// write the scene properties to the vertex shader argument buffer at index 0
let scene_properties_bytes = NonNull::from(scene_properties_data);
Expand Down
9 changes: 9 additions & 0 deletions crates/icrate/src/fixes/Foundation/copying.rs
Original file line number Diff line number Diff line change
Expand Up @@ -34,6 +34,15 @@ extern_protocol!(
/// # Safety
///
/// The zone pointer must be valid or NULL.
///
///
/// # Examples
///
/// Examples on how to implement copyWithZone:
///
/// TODO
///
/// The different ways described in apples doc
#[method_id(copyWithZone:)]
unsafe fn copyWithZone(&self, zone: *mut NSZone) -> Id<Self::Immutable>
where
Expand Down
4 changes: 2 additions & 2 deletions crates/icrate/tests/array.rs
Original file line number Diff line number Diff line change
Expand Up @@ -122,7 +122,7 @@ fn test_retains_stored() {

drop(obj);
expected.release += 1;
expected.dealloc += 1;
expected.drop += 1;
expected.assert_current();
}

Expand Down Expand Up @@ -187,7 +187,7 @@ fn test_iter_minimal_retains() {

assert_eq!(iter.count(), 0);
expected.release += 1;
expected.dealloc += 1;
expected.drop += 1;
expected.assert_current();
}

Expand Down
4 changes: 3 additions & 1 deletion crates/icrate/tests/auto_traits.rs
Original file line number Diff line number Diff line change
Expand Up @@ -7,7 +7,7 @@ use icrate::Foundation::*;
use objc2::mutability::{Immutable, Mutable};
use objc2::rc::Id;
use objc2::runtime::AnyObject;
use objc2::{declare_class, ClassType};
use objc2::{declare_class, ClassType, DeclaredClass};

// We expect most Foundation types to be UnwindSafe and RefUnwindSafe,
// since they follow Rust's usual mutability rules (&T = immutable).
Expand Down Expand Up @@ -40,6 +40,8 @@ macro_rules! helper {
type Mutability = $mutability;
const NAME: &'static str = concat!(stringify!($name), "Test");
}

impl DeclaredClass for $name {}
);
};
}
Expand Down
2 changes: 1 addition & 1 deletion crates/icrate/tests/mutable_array.rs
Original file line number Diff line number Diff line change
Expand Up @@ -151,7 +151,7 @@ fn test_remove() {

array.removeAllObjects();
expected.release += 2;
expected.dealloc += 2;
expected.drop += 2;
expected.assert_current();
assert_eq!(array.len(), 0);
}
Expand Down
6 changes: 3 additions & 3 deletions crates/icrate/tests/mutable_dictionary.rs
Original file line number Diff line number Diff line change
Expand Up @@ -102,7 +102,7 @@ fn test_insert_key_copies() {
dict.removeAllObjects();
// Release key
expected.release += 1;
expected.dealloc += 1;
expected.drop += 1;
expected.assert_current();
}

Expand Down Expand Up @@ -137,7 +137,7 @@ fn test_insert_value_retain_release() {

drop(old);
expected.release += 1;
expected.dealloc += 1;
expected.drop += 1;
expected.assert_current();
}

Expand Down Expand Up @@ -183,7 +183,7 @@ fn test_remove_clear_release_dealloc() {

dict.removeAllObjects();
expected.release += 2;
expected.dealloc += 2;
expected.drop += 2;
expected.assert_current();
assert_eq!(dict.len(), 0);
}
Expand Down
2 changes: 1 addition & 1 deletion crates/icrate/tests/mutable_set.rs
Original file line number Diff line number Diff line change
Expand Up @@ -113,7 +113,7 @@ fn test_clear_release_dealloc() {

set.removeAllObjects();
expected.release += 4;
expected.dealloc += 4;
expected.drop += 4;
expected.assert_current();
assert_eq!(set.len(), 0);
}
4 changes: 2 additions & 2 deletions crates/icrate/tests/set.rs
Original file line number Diff line number Diff line change
Expand Up @@ -227,7 +227,7 @@ fn test_retains_stored() {

drop(obj);
expected.release += 1;
expected.dealloc += 1;
expected.drop += 1;
expected.assert_current();
}

Expand Down Expand Up @@ -291,7 +291,7 @@ fn test_iter_minimal_retains() {

assert_eq!(iter.count(), 0);
expected.release += 1;
expected.dealloc += 1;
expected.drop += 1;
expected.assert_current();
}

Expand Down
Loading

0 comments on commit 42dac6d

Please sign in to comment.