mirror of
https://github.com/italicsjenga/rust_minifb.git
synced 2025-01-26 02:36:32 +11:00
WIP on Mac version
This commit is contained in:
parent
026be10b8b
commit
1455bd375e
1 changed files with 100 additions and 26 deletions
126
src/macos.rs
126
src/macos.rs
|
@ -17,7 +17,7 @@ use objc::declare::ClassDecl;
|
||||||
|
|
||||||
#[allow(unused_imports)]
|
#[allow(unused_imports)]
|
||||||
use cocoa::foundation::{NSAutoreleasePool, NSDate, NSDefaultRunLoopMode, NSPoint, NSRect, NSSize,
|
use cocoa::foundation::{NSAutoreleasePool, NSDate, NSDefaultRunLoopMode, NSPoint, NSRect, NSSize,
|
||||||
NSString, NSUInteger};
|
NSString, NSUInteger};
|
||||||
|
|
||||||
use std::ops::Deref;
|
use std::ops::Deref;
|
||||||
|
|
||||||
|
@ -25,6 +25,7 @@ pub struct Window {
|
||||||
view: IdRef,
|
view: IdRef,
|
||||||
window: IdRef,
|
window: IdRef,
|
||||||
delegate: WindowDelegate,
|
delegate: WindowDelegate,
|
||||||
|
view_delegate: ViewDelegate,
|
||||||
}
|
}
|
||||||
|
|
||||||
struct DelegateState {
|
struct DelegateState {
|
||||||
|
@ -38,7 +39,14 @@ struct WindowDelegate {
|
||||||
_this: IdRef,
|
_this: IdRef,
|
||||||
}
|
}
|
||||||
|
|
||||||
//sthou
|
struct ViewDelegateState {
|
||||||
|
view: IdRef,
|
||||||
|
}
|
||||||
|
|
||||||
|
struct ViewDelegate {
|
||||||
|
state: Box<ViewDelegateState>,
|
||||||
|
_this: IdRef,
|
||||||
|
}
|
||||||
|
|
||||||
impl WindowDelegate {
|
impl WindowDelegate {
|
||||||
/// Get the delegate class, initiailizing it neccessary
|
/// Get the delegate class, initiailizing it neccessary
|
||||||
|
@ -46,9 +54,10 @@ impl WindowDelegate {
|
||||||
use std::sync::{Once, ONCE_INIT};
|
use std::sync::{Once, ONCE_INIT};
|
||||||
|
|
||||||
extern fn window_should_close(this: &Object, _: Sel, _: id) -> BOOL {
|
extern fn window_should_close(this: &Object, _: Sel, _: id) -> BOOL {
|
||||||
|
println!("Should close");
|
||||||
unsafe {
|
unsafe {
|
||||||
let state: *mut libc::c_void = *this.get_ivar("glutinState");
|
//let state: *mut libc::c_void = *this.get_ivar("minifb_window_state");
|
||||||
let state = state as *mut DelegateState;
|
//let state = state as *mut DelegateState;
|
||||||
//(*state).pending_events.lock().unwrap().push_back(Closed);
|
//(*state).pending_events.lock().unwrap().push_back(Closed);
|
||||||
}
|
}
|
||||||
YES
|
YES
|
||||||
|
@ -56,7 +65,7 @@ impl WindowDelegate {
|
||||||
|
|
||||||
extern fn window_did_resize(this: &Object, _: Sel, _: id) {
|
extern fn window_did_resize(this: &Object, _: Sel, _: id) {
|
||||||
unsafe {
|
unsafe {
|
||||||
let state: *mut libc::c_void = *this.get_ivar("glutinState");
|
let state: *mut libc::c_void = *this.get_ivar("minifb_window_state");
|
||||||
let state = &mut *(state as *mut DelegateState);
|
let state = &mut *(state as *mut DelegateState);
|
||||||
|
|
||||||
println!("did_resize");
|
println!("did_resize");
|
||||||
|
@ -67,7 +76,7 @@ impl WindowDelegate {
|
||||||
let rect = NSView::frame(*state.view);
|
let rect = NSView::frame(*state.view);
|
||||||
let scale_factor = NSWindow::backingScaleFactor(*state.window) as f32;
|
let scale_factor = NSWindow::backingScaleFactor(*state.window) as f32;
|
||||||
(handler)((scale_factor * rect.size.width as f32) as u32,
|
(handler)((scale_factor * rect.size.width as f32) as u32,
|
||||||
(scale_factor * rect.size.height as f32) as u32);
|
(scale_factor * rect.size.height as f32) as u32);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -78,7 +87,7 @@ impl WindowDelegate {
|
||||||
// lost focus
|
// lost focus
|
||||||
|
|
||||||
println!("became key window");
|
println!("became key window");
|
||||||
let state: *mut libc::c_void = *this.get_ivar("glutinState");
|
let state: *mut libc::c_void = *this.get_ivar("minifb_window_state");
|
||||||
let state = state as *mut DelegateState;
|
let state = state as *mut DelegateState;
|
||||||
//(*state).pending_events.lock().unwrap().push_back(Focused(true));
|
//(*state).pending_events.lock().unwrap().push_back(Focused(true));
|
||||||
}
|
}
|
||||||
|
@ -86,7 +95,7 @@ impl WindowDelegate {
|
||||||
|
|
||||||
extern fn window_did_resign_key(this: &Object, _: Sel, _: id) {
|
extern fn window_did_resign_key(this: &Object, _: Sel, _: id) {
|
||||||
unsafe {
|
unsafe {
|
||||||
let state: *mut libc::c_void = *this.get_ivar("glutinState");
|
let state: *mut libc::c_void = *this.get_ivar("minifb_window_state");
|
||||||
let state = state as *mut DelegateState;
|
let state = state as *mut DelegateState;
|
||||||
//(*state).pending_events.lock().unwrap().push_back(Focused(false));
|
//(*state).pending_events.lock().unwrap().push_back(Focused(false));
|
||||||
}
|
}
|
||||||
|
@ -98,21 +107,21 @@ impl WindowDelegate {
|
||||||
INIT.call_once(|| unsafe {
|
INIT.call_once(|| unsafe {
|
||||||
println!("Create new NSWindowDelegate");
|
println!("Create new NSWindowDelegate");
|
||||||
let superclass = Class::get("NSObject").unwrap();
|
let superclass = Class::get("NSObject").unwrap();
|
||||||
let mut decl = ClassDecl::new(superclass, "GlutinWindowDelegate").unwrap();
|
let mut decl = ClassDecl::new(superclass, "minifb_window_delegate").unwrap();
|
||||||
|
|
||||||
// Add callback methods
|
// Add callback methods
|
||||||
decl.add_method(sel!(windowShouldClose:),
|
decl.add_method(sel!(windowShouldClose:),
|
||||||
window_should_close as extern fn(&Object, Sel, id) -> BOOL);
|
window_should_close as extern fn(&Object, Sel, id) -> BOOL);
|
||||||
decl.add_method(sel!(windowDidResize:),
|
decl.add_method(sel!(windowDidResize:),
|
||||||
window_did_resize as extern fn(&Object, Sel, id));
|
window_did_resize as extern fn(&Object, Sel, id));
|
||||||
|
|
||||||
decl.add_method(sel!(windowDidBecomeKey:),
|
decl.add_method(sel!(windowDidBecomeKey:),
|
||||||
window_did_become_key as extern fn(&Object, Sel, id));
|
window_did_become_key as extern fn(&Object, Sel, id));
|
||||||
decl.add_method(sel!(windowDidResignKey:),
|
decl.add_method(sel!(windowDidResignKey:),
|
||||||
window_did_resign_key as extern fn(&Object, Sel, id));
|
window_did_resign_key as extern fn(&Object, Sel, id));
|
||||||
|
|
||||||
// Store internal state as user data
|
// Store internal state as user data
|
||||||
decl.add_ivar::<*mut libc::c_void>("glutinState");
|
decl.add_ivar::<*mut libc::c_void>("minifb_window_state");
|
||||||
|
|
||||||
delegate_class = decl.register();
|
delegate_class = decl.register();
|
||||||
});
|
});
|
||||||
|
@ -129,9 +138,11 @@ impl WindowDelegate {
|
||||||
unsafe {
|
unsafe {
|
||||||
let delegate = IdRef::new(msg_send![WindowDelegate::class(), new]);
|
let delegate = IdRef::new(msg_send![WindowDelegate::class(), new]);
|
||||||
|
|
||||||
(&mut **delegate).set_ivar("glutinState", state_ptr as *mut libc::c_void);
|
(&mut **delegate).set_ivar("minifb_window_state", state_ptr as *mut libc::c_void);
|
||||||
let _: () = msg_send![*state.window, setDelegate:*delegate];
|
let _: () = msg_send![*state.window, setDelegate:*delegate];
|
||||||
|
|
||||||
|
println!("Setup delegate");
|
||||||
|
|
||||||
WindowDelegate { state: state, _this: delegate }
|
WindowDelegate { state: state, _this: delegate }
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -146,6 +157,60 @@ impl Drop for WindowDelegate {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
impl ViewDelegate {
|
||||||
|
fn class() -> *const Class {
|
||||||
|
use std::sync::{Once, ONCE_INIT};
|
||||||
|
|
||||||
|
extern fn draw_rect(this: &Object, _: Sel, _: id) {
|
||||||
|
println!("draw_rect");
|
||||||
|
}
|
||||||
|
|
||||||
|
static mut delegate_class: *const Class = 0 as *const Class;
|
||||||
|
static INIT: Once = ONCE_INIT;
|
||||||
|
|
||||||
|
INIT.call_once(|| unsafe {
|
||||||
|
println!("Create new ViewDelegate");
|
||||||
|
let superclass = Class::get("NSObject").unwrap();
|
||||||
|
let mut decl = ClassDecl::new(superclass, "minifb_view_delegate").unwrap();
|
||||||
|
|
||||||
|
// Add callback methods
|
||||||
|
decl.add_method(sel!(drawRect:),
|
||||||
|
draw_rect as extern fn(&Object, Sel, id));
|
||||||
|
// Store internal state as user data
|
||||||
|
decl.add_ivar::<*mut libc::c_void>("minifb_view_state");
|
||||||
|
|
||||||
|
delegate_class = decl.register();
|
||||||
|
});
|
||||||
|
|
||||||
|
unsafe {
|
||||||
|
delegate_class
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn new(state: ViewDelegateState) -> ViewDelegate {
|
||||||
|
let mut state = Box::new(state);
|
||||||
|
let state_ptr: *mut ViewDelegateState = &mut *state;
|
||||||
|
unsafe {
|
||||||
|
let delegate = IdRef::new(msg_send![ViewDelegate::class(), new]);
|
||||||
|
|
||||||
|
(&mut **delegate).set_ivar("minifb_view_state", state_ptr as *mut libc::c_void);
|
||||||
|
let _: () = msg_send![*state.view, setDelegate:*delegate];
|
||||||
|
|
||||||
|
ViewDelegate { state: state, _this: delegate }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Drop for ViewDelegate {
|
||||||
|
fn drop(&mut self) {
|
||||||
|
unsafe {
|
||||||
|
// Nil the views's delegate so it doesn't still reference us
|
||||||
|
let _: () = msg_send![*self.state.view, setDelegate:nil];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
impl Window {
|
impl Window {
|
||||||
pub fn new(name: &str, width: usize, height: usize, _: Scale, _: Vsync) -> Result<Window, &str> {
|
pub fn new(name: &str, width: usize, height: usize, _: Scale, _: Vsync) -> Result<Window, &str> {
|
||||||
|
@ -170,7 +235,7 @@ impl Window {
|
||||||
return Err("Unable to create NSView");
|
return Err("Unable to create NSView");
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
app.activateIgnoringOtherApps_(YES);
|
app.activateIgnoringOtherApps_(YES);
|
||||||
window.makeKeyAndOrderFront_(nil);
|
window.makeKeyAndOrderFront_(nil);
|
||||||
|
|
||||||
|
@ -180,29 +245,34 @@ impl Window {
|
||||||
resize_handler: None,
|
resize_handler: None,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
let vs = ViewDelegateState {
|
||||||
|
view: view.clone(),
|
||||||
|
};
|
||||||
|
|
||||||
println!("Created window and view");
|
println!("Created window and view");
|
||||||
|
|
||||||
return Ok(Window {
|
return Ok(Window {
|
||||||
window: window,
|
window: window,
|
||||||
view: view,
|
view: view,
|
||||||
delegate: WindowDelegate::new(ds),
|
delegate: WindowDelegate::new(ds),
|
||||||
|
view_delegate: ViewDelegate::new(vs),
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
unsafe fn create_window(name: &str, width: usize, height: usize) -> Option<IdRef> {
|
unsafe fn create_window(name: &str, width: usize, height: usize) -> Option<IdRef> {
|
||||||
let masks = NSResizableWindowMask as NSUInteger |
|
let masks = NSResizableWindowMask as NSUInteger |
|
||||||
NSClosableWindowMask as NSUInteger |
|
NSClosableWindowMask as NSUInteger |
|
||||||
NSTitledWindowMask as NSUInteger;
|
NSTitledWindowMask as NSUInteger;
|
||||||
|
|
||||||
let frame = NSRect::new(NSPoint::new(0., 0.), NSSize::new(width as f64, height as f64));
|
let frame = NSRect::new(NSPoint::new(0., 0.), NSSize::new(width as f64, height as f64));
|
||||||
|
|
||||||
let window = IdRef::new(NSWindow::alloc(nil).initWithContentRect_styleMask_backing_defer_(
|
let window = IdRef::new(NSWindow::alloc(nil).initWithContentRect_styleMask_backing_defer_(
|
||||||
frame,
|
frame,
|
||||||
masks,
|
masks,
|
||||||
NSBackingStoreBuffered,
|
NSBackingStoreBuffered,
|
||||||
NO,
|
NO,
|
||||||
));
|
));
|
||||||
|
|
||||||
window.non_nil().map(|window| {
|
window.non_nil().map(|window| {
|
||||||
let title = IdRef::new(NSString::alloc(nil).init_str(name));
|
let title = IdRef::new(NSString::alloc(nil).init_str(name));
|
||||||
|
@ -223,11 +293,15 @@ impl Window {
|
||||||
|
|
||||||
pub fn update(&self, _: &[u32]) {
|
pub fn update(&self, _: &[u32]) {
|
||||||
unsafe {
|
unsafe {
|
||||||
NSApp().nextEventMatchingMask_untilDate_inMode_dequeue_(
|
let event = NSApp().nextEventMatchingMask_untilDate_inMode_dequeue_(
|
||||||
NSAnyEventMask.bits(),
|
NSAnyEventMask.bits(),
|
||||||
NSDate::distantPast(nil),
|
NSDate::distantPast(nil),
|
||||||
NSDefaultRunLoopMode,
|
NSDefaultRunLoopMode,
|
||||||
YES);
|
YES);
|
||||||
|
|
||||||
|
if event != nil {
|
||||||
|
NSApp().sendEvent_(event);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Add table
Reference in a new issue