From c38110cac77253761afc8cc616f645dc1ec4d3b8 Mon Sep 17 00:00:00 2001 From: David Partouche Date: Sat, 4 Oct 2014 15:49:39 +0200 Subject: Cocoa opens a window with OpenGL context inside --- src/osx/mod.rs | 106 ++++++++++++++++++++++++++++++++++++++++++++++++++------- 1 file changed, 94 insertions(+), 12 deletions(-) (limited to 'src/osx/mod.rs') diff --git a/src/osx/mod.rs b/src/osx/mod.rs index 6960cc1..99a8b2b 100644 --- a/src/osx/mod.rs +++ b/src/osx/mod.rs @@ -1,12 +1,17 @@ -//! Dummy implementation for OS/X to make gl-init-rs compile on this platform - use {Event, WindowBuilder}; -pub struct Window; +use cocoa::base::{id, NSUInteger, nil}; +use cocoa::appkit::*; -pub struct MonitorID; +use core_foundation::base::TCFType; +use core_foundation::string::CFString; +use core_foundation::bundle::{CFBundleGetBundleWithIdentifier, CFBundleGetFunctionPointerForName}; + +pub struct Window { + context: id, +} -compile_warning!("The OS/X platform is not implemented yet") +pub struct MonitorID; pub fn get_available_monitors() -> Vec { unimplemented!() @@ -18,7 +23,7 @@ pub fn get_primary_monitor() -> MonitorID { impl MonitorID { pub fn get_name(&self) -> Option { - unimplemented!() + unimplemented!() } pub fn get_dimensions(&self) -> (uint, uint) { @@ -28,11 +33,68 @@ impl MonitorID { impl Window { pub fn new(_builder: WindowBuilder) -> Result { - unimplemented!() + let context = unsafe { + // Create the NSApplication + let app = NSApp(); + app.setActivationPolicy_(NSApplicationActivationPolicyRegular); + + app.finishLaunching(); + + // Create the window + let scr_frame = NSRect::new(NSPoint::new(0., 0.), NSSize::new(800., 600.)); + + let window = NSWindow::alloc(nil).initWithContentRect_styleMask_backing_defer_( + scr_frame, + NSTitledWindowMask as NSUInteger | NSClosableWindowMask as NSUInteger | NSMiniaturizableWindowMask as NSUInteger, + NSBackingStoreBuffered, + false + ); + let view = NSView::alloc(nil).init(); + view.setWantsBestResolutionOpenGLSurface_(true); + + let title = NSString::alloc(nil).init_str("Hello World!\0"); + window.setTitle_(title); + window.setContentView(view); + window.center(); + + // Create the context + let attributes = [ + NSOpenGLPFADoubleBuffer as uint, + NSOpenGLPFAClosestPolicy as uint, + NSOpenGLPFAColorSize as uint, 24, + NSOpenGLPFAAlphaSize as uint, 8, + NSOpenGLPFADepthSize as uint, 24, + NSOpenGLPFAStencilSize as uint, 8, + 0 + ]; + + let pixelformat = NSOpenGLPixelFormat::alloc(nil).initWithAttributes_(attributes); + if pixelformat == nil { + return Err(format!("Couldn't create the pixel format")); + } + + let context = NSOpenGLContext::alloc(nil).initWithFormat_shareContext_(pixelformat, nil); + if context == nil { + return Err(format!("No valid OpenGL context can be created with that pixelformat")); + } + + context.setView_(view); + + app.activateIgnoringOtherApps_(true); + window.makeKeyAndOrderFront_(nil); + context + }; + + let window = Window { + context: context, + }; + + Ok(window) } pub fn is_closed(&self) -> bool { - unimplemented!() + // TODO: remove fake implementation + false } pub fn set_title(&self, _title: &str) { @@ -64,18 +126,38 @@ impl Window { } pub fn wait_events(&self) -> Vec { - unimplemented!() + loop { + unsafe { + let event = NSApp().nextEventMatchingMask_untilDate_inMode_dequeue_( + NSAnyEventMask as u64, + nil, + NSDefaultRunLoopMode, + true); + if event == nil { break; } + NSApp().sendEvent_(event); + } + } + // TODO: Remove fake implementation + Vec::new() } pub unsafe fn make_current(&self) { - unimplemented!() + self.context.makeCurrentContext(); } pub fn get_proc_address(&self, _addr: &str) -> *const () { - unimplemented!() + let symbol_name: CFString = from_str(_addr).unwrap(); + let framework_name: CFString = from_str("com.apple.opengl").unwrap(); + let framework = unsafe { + CFBundleGetBundleWithIdentifier(framework_name.as_concrete_TypeRef()) + }; + let symbol = unsafe { + CFBundleGetFunctionPointerForName(framework, symbol_name.as_concrete_TypeRef()) + }; + symbol as *const () } pub fn swap_buffers(&self) { - unimplemented!() + unsafe { self.context.flushBuffer(); } } } -- cgit v1.2.3 From db578e6e1ce88f2ceb2f0f04315aedab9e11441e Mon Sep 17 00:00:00 2001 From: David Partouche Date: Sun, 5 Oct 2014 23:45:38 +0200 Subject: Refactored window initialisation code for cocoa, making it more memory safe --- Cargo.toml | 2 +- src/osx/mod.rs | 106 +++++++++++++++++++++++++++++++++++++++++---------------- 2 files changed, 78 insertions(+), 30 deletions(-) (limited to 'src/osx/mod.rs') diff --git a/Cargo.toml b/Cargo.toml index 1df2220..d15ba32 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -17,4 +17,4 @@ git = "https://github.com/bjz/gl-rs" git = "https://github.com/DavidPartouche/rust-cocoa" [dependencies.core_foundation] -git = "https://github.com/DavidPartouche/rust-core-foundation" +git = "https://github.com/servo/rust-core-foundation" diff --git a/src/osx/mod.rs b/src/osx/mod.rs index 99a8b2b..1dbe383 100644 --- a/src/osx/mod.rs +++ b/src/osx/mod.rs @@ -33,15 +33,54 @@ impl MonitorID { impl Window { pub fn new(_builder: WindowBuilder) -> Result { - let context = unsafe { - // Create the NSApplication - let app = NSApp(); - app.setActivationPolicy_(NSApplicationActivationPolicyRegular); - app.finishLaunching(); + let app = match Window::create_app() { + Some(app) => app, + None => { return Err(format!("Couldn't create NSApplication")); }, + }; + let window = match Window::create_window(_builder.dimensions.unwrap_or((800, 600)), _builder.title.as_slice()) { + Some(window) => window, + None => { return Err(format!("Couldn't create NSWindow")); }, + }; + let view = match Window::create_view(window) { + Some(view) => view, + None => { return Err(format!("Couldn't create NSView")); }, + }; + + let context = match Window::create_context(view) { + Some(context) => context, + None => { return Err(format!("Couldn't create OpenGL context")); }, + }; + + unsafe { + app.activateIgnoringOtherApps_(true); + window.makeKeyAndOrderFront_(nil); + } + + let window = Window { + context: context, + }; + + Ok(window) + } + + fn create_app() -> Option { + unsafe { + let app = NSApp(); + if app == nil { + None + } else { + app.setActivationPolicy_(NSApplicationActivationPolicyRegular); + app.finishLaunching(); + Some(app) + } + } + } - // Create the window - let scr_frame = NSRect::new(NSPoint::new(0., 0.), NSSize::new(800., 600.)); + fn create_window(dimensions: (uint, uint), title: &str) -> Option { + unsafe { + let (width, height) = dimensions; + let scr_frame = NSRect::new(NSPoint::new(0., 0.), NSSize::new(width as f64, height as f64)); let window = NSWindow::alloc(nil).initWithContentRect_styleMask_backing_defer_( scr_frame, @@ -49,15 +88,33 @@ impl Window { NSBackingStoreBuffered, false ); - let view = NSView::alloc(nil).init(); - view.setWantsBestResolutionOpenGLSurface_(true); - let title = NSString::alloc(nil).init_str("Hello World!\0"); - window.setTitle_(title); - window.setContentView(view); - window.center(); + if window == nil { + None + } else { + let title = NSString::alloc(nil).init_str(title); + window.setTitle_(title); + window.center(); + Some(window) + } + } + } - // Create the context + fn create_view(window: id) -> Option { + unsafe { + let view = NSView::alloc(nil).init(); + if view == nil { + None + } else { + view.setWantsBestResolutionOpenGLSurface_(true); + window.setContentView(view); + Some(view) + } + } + } + + fn create_context(view: id) -> Option { + unsafe { let attributes = [ NSOpenGLPFADoubleBuffer as uint, NSOpenGLPFAClosestPolicy as uint, @@ -70,26 +127,17 @@ impl Window { let pixelformat = NSOpenGLPixelFormat::alloc(nil).initWithAttributes_(attributes); if pixelformat == nil { - return Err(format!("Couldn't create the pixel format")); + return None; } let context = NSOpenGLContext::alloc(nil).initWithFormat_shareContext_(pixelformat, nil); if context == nil { - return Err(format!("No valid OpenGL context can be created with that pixelformat")); + None + } else { + context.setView_(view); + Some(context) } - - context.setView_(view); - - app.activateIgnoringOtherApps_(true); - window.makeKeyAndOrderFront_(nil); - context - }; - - let window = Window { - context: context, - }; - - Ok(window) + } } pub fn is_closed(&self) -> bool { -- cgit v1.2.3 From 989fd1b505a51bde3f8bcd2bf9b94fa434880dd6 Mon Sep 17 00:00:00 2001 From: David Partouche Date: Mon, 6 Oct 2014 17:54:15 +0200 Subject: Added events handling for left/right mouse clicking --- src/osx/mod.rs | 39 ++++++++++++++++++++++++++++++++------- 1 file changed, 32 insertions(+), 7 deletions(-) (limited to 'src/osx/mod.rs') diff --git a/src/osx/mod.rs b/src/osx/mod.rs index 1dbe383..e61e536 100644 --- a/src/osx/mod.rs +++ b/src/osx/mod.rs @@ -170,23 +170,48 @@ impl Window { } pub fn poll_events(&self) -> Vec { - unimplemented!() - } + let mut events = Vec::new(); - pub fn wait_events(&self) -> Vec { loop { unsafe { + use {MouseInput, Pressed, Released, LeftMouseButton, RightMouseButton}; let event = NSApp().nextEventMatchingMask_untilDate_inMode_dequeue_( NSAnyEventMask as u64, - nil, + NSDate::distantPast(nil), NSDefaultRunLoopMode, true); if event == nil { break; } - NSApp().sendEvent_(event); + + match event.get_type() { + NSLeftMouseDown => { events.push(MouseInput(Pressed, LeftMouseButton)); }, + NSLeftMouseUp => { events.push(MouseInput(Released, LeftMouseButton)); }, + NSRightMouseDown => { events.push(MouseInput(Pressed, RightMouseButton)); }, + NSRightMouseUp => { events.push(MouseInput(Released, RightMouseButton)); }, + NSMouseMoved => { }, + NSKeyDown => { }, + NSKeyUp => { }, + NSFlagsChanged => { }, + NSScrollWheel => { }, + NSOtherMouseDown => { }, + NSOtherMouseUp => { }, + NSOtherMouseDragged => { }, + _ => { }, + } } } - // TODO: Remove fake implementation - Vec::new() + events + } + + pub fn wait_events(&self) -> Vec { + unsafe { + let event = NSApp().nextEventMatchingMask_untilDate_inMode_dequeue_( + NSAnyEventMask as u64, + NSDate::distantFuture(nil), + NSDefaultRunLoopMode, + true); + NSApp().sendEvent_(event); + self.poll_events() + } } pub unsafe fn make_current(&self) { -- cgit v1.2.3 From dc80f57bc27f430d94b14470d3c0fded6da5b185 Mon Sep 17 00:00:00 2001 From: DavidPartouche Date: Mon, 6 Oct 2014 18:37:06 +0200 Subject: Fixed a space missing --- src/osx/mod.rs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) (limited to 'src/osx/mod.rs') diff --git a/src/osx/mod.rs b/src/osx/mod.rs index e61e536..d41c9c7 100644 --- a/src/osx/mod.rs +++ b/src/osx/mod.rs @@ -23,7 +23,7 @@ pub fn get_primary_monitor() -> MonitorID { impl MonitorID { pub fn get_name(&self) -> Option { - unimplemented!() + unimplemented!() } pub fn get_dimensions(&self) -> (uint, uint) { -- cgit v1.2.3