From 959831f4ef5a3e797f576c3de08cd65032c997ad Mon Sep 17 00:00:00 2001 From: David Walter Seikel Date: Sun, 13 Jan 2013 18:54:10 +1000 Subject: Remove damned ancient DOS line endings from Irrlicht. Hopefully I did not go overboard. --- .../source/Irrlicht/MacOSX/AppDelegate.h | 48 +- .../source/Irrlicht/MacOSX/AppDelegate.mm | 158 +- .../source/Irrlicht/MacOSX/CIrrDeviceMacOSX.h | 502 +-- .../source/Irrlicht/MacOSX/CIrrDeviceMacOSX.mm | 3784 ++++++++++---------- .../source/Irrlicht/MacOSX/MacOSX_Prefix.pch | 10 +- .../Irrlicht/MacOSX/MainMenu.nib/keyedobjects.nib | Bin 13030 -> 13029 bytes .../source/Irrlicht/MacOSX/OSXClipboard.h | 30 +- .../source/Irrlicht/MacOSX/OSXClipboard.mm | 72 +- 8 files changed, 2302 insertions(+), 2302 deletions(-) (limited to 'libraries/irrlicht-1.8/source/Irrlicht/MacOSX') diff --git a/libraries/irrlicht-1.8/source/Irrlicht/MacOSX/AppDelegate.h b/libraries/irrlicht-1.8/source/Irrlicht/MacOSX/AppDelegate.h index ccb116d..4a821db 100644 --- a/libraries/irrlicht-1.8/source/Irrlicht/MacOSX/AppDelegate.h +++ b/libraries/irrlicht-1.8/source/Irrlicht/MacOSX/AppDelegate.h @@ -1,24 +1,24 @@ -// Copyright (C) 2005-2006 Etienne Petitjean -// Copyright (C) 2007-2012 Christian Stehno -// This file is part of the "Irrlicht Engine". -// For conditions of distribution and use, see copyright notice in Irrlicht.h - -#include "IrrCompileConfig.h" - -#ifdef _IRR_COMPILE_WITH_OSX_DEVICE_ - -#import -#import "CIrrDeviceMacOSX.h" - -@interface AppDelegate : NSObject -{ - BOOL _quit; - irr::CIrrDeviceMacOSX *_device; -} - -- (id)initWithDevice:(irr::CIrrDeviceMacOSX *)device; -- (BOOL)isQuit; - -@end - -#endif // _IRR_COMPILE_WITH_OSX_DEVICE_ +// Copyright (C) 2005-2006 Etienne Petitjean +// Copyright (C) 2007-2012 Christian Stehno +// This file is part of the "Irrlicht Engine". +// For conditions of distribution and use, see copyright notice in Irrlicht.h + +#include "IrrCompileConfig.h" + +#ifdef _IRR_COMPILE_WITH_OSX_DEVICE_ + +#import +#import "CIrrDeviceMacOSX.h" + +@interface AppDelegate : NSObject +{ + BOOL _quit; + irr::CIrrDeviceMacOSX *_device; +} + +- (id)initWithDevice:(irr::CIrrDeviceMacOSX *)device; +- (BOOL)isQuit; + +@end + +#endif // _IRR_COMPILE_WITH_OSX_DEVICE_ diff --git a/libraries/irrlicht-1.8/source/Irrlicht/MacOSX/AppDelegate.mm b/libraries/irrlicht-1.8/source/Irrlicht/MacOSX/AppDelegate.mm index 14a7f86..a8b2915 100644 --- a/libraries/irrlicht-1.8/source/Irrlicht/MacOSX/AppDelegate.mm +++ b/libraries/irrlicht-1.8/source/Irrlicht/MacOSX/AppDelegate.mm @@ -1,79 +1,79 @@ -// Copyright (C) 2005-2006 Etienne Petitjean -// Copyright (C) 2007-2012 Christian Stehno -// This file is part of the "Irrlicht Engine". -// For conditions of distribution and use, see copyright notice in Irrlicht.h - -#import "AppDelegate.h" - -#ifdef _IRR_COMPILE_WITH_OSX_DEVICE_ - -@implementation AppDelegate - -- (id)initWithDevice:(irr::CIrrDeviceMacOSX *)device -{ - self = [super init]; - if (self) _device = device; - return (self); -} - -- (void)applicationDidFinishLaunching:(NSNotification *)aNotification -{ - _quit = FALSE; -} - -- (void)orderFrontStandardAboutPanel:(id)sender -{ - [NSApp orderFrontStandardAboutPanel:sender]; -} - -- (void)unhideAllApplications:(id)sender -{ - [NSApp unhideAllApplications:sender]; -} - -- (void)hide:(id)sender -{ - [NSApp hide:sender]; -} - -- (void)hideOtherApplications:(id)sender -{ - [NSApp hideOtherApplications:sender]; -} - -- (void)terminate:(id)sender -{ - _quit = TRUE; -} - -- (void)windowWillClose:(id)sender -{ - _quit = TRUE; -} - -- (NSSize)windowWillResize:(NSWindow *)window toSize:(NSSize)proposedFrameSize -{ - if (_device->isResizable()) - return proposedFrameSize; - else - return [window frame].size; -} - -- (void)windowDidResize:(NSNotification *)aNotification -{ - NSWindow *window; - NSRect frame; - - window = [aNotification object]; - frame = [window frame]; - _device->setResize((int)frame.size.width,(int)frame.size.height); -} - -- (BOOL)isQuit -{ - return (_quit); -} - -@end - -#endif // _IRR_COMPILE_WITH_OSX_DEVICE_ +// Copyright (C) 2005-2006 Etienne Petitjean +// Copyright (C) 2007-2012 Christian Stehno +// This file is part of the "Irrlicht Engine". +// For conditions of distribution and use, see copyright notice in Irrlicht.h + +#import "AppDelegate.h" + +#ifdef _IRR_COMPILE_WITH_OSX_DEVICE_ + +@implementation AppDelegate + +- (id)initWithDevice:(irr::CIrrDeviceMacOSX *)device +{ + self = [super init]; + if (self) _device = device; + return (self); +} + +- (void)applicationDidFinishLaunching:(NSNotification *)aNotification +{ + _quit = FALSE; +} + +- (void)orderFrontStandardAboutPanel:(id)sender +{ + [NSApp orderFrontStandardAboutPanel:sender]; +} + +- (void)unhideAllApplications:(id)sender +{ + [NSApp unhideAllApplications:sender]; +} + +- (void)hide:(id)sender +{ + [NSApp hide:sender]; +} + +- (void)hideOtherApplications:(id)sender +{ + [NSApp hideOtherApplications:sender]; +} + +- (void)terminate:(id)sender +{ + _quit = TRUE; +} + +- (void)windowWillClose:(id)sender +{ + _quit = TRUE; +} + +- (NSSize)windowWillResize:(NSWindow *)window toSize:(NSSize)proposedFrameSize +{ + if (_device->isResizable()) + return proposedFrameSize; + else + return [window frame].size; +} + +- (void)windowDidResize:(NSNotification *)aNotification +{ + NSWindow *window; + NSRect frame; + + window = [aNotification object]; + frame = [window frame]; + _device->setResize((int)frame.size.width,(int)frame.size.height); +} + +- (BOOL)isQuit +{ + return (_quit); +} + +@end + +#endif // _IRR_COMPILE_WITH_OSX_DEVICE_ diff --git a/libraries/irrlicht-1.8/source/Irrlicht/MacOSX/CIrrDeviceMacOSX.h b/libraries/irrlicht-1.8/source/Irrlicht/MacOSX/CIrrDeviceMacOSX.h index 174c68f..5b8ab9a 100644 --- a/libraries/irrlicht-1.8/source/Irrlicht/MacOSX/CIrrDeviceMacOSX.h +++ b/libraries/irrlicht-1.8/source/Irrlicht/MacOSX/CIrrDeviceMacOSX.h @@ -1,251 +1,251 @@ -// Copyright (C) 2005-2006 Etienne Petitjean -// Copyright (C) 2007-2012 Christian Stehno -// This file is part of the "Irrlicht Engine". -// For conditions of distribution and use, see copyright notice in Irrlicht.h - -#ifndef __C_IRR_DEVICE_MACOSX_H_INCLUDED__ -#define __C_IRR_DEVICE_MACOSX_H_INCLUDED__ - -#include "IrrCompileConfig.h" - -#ifdef _IRR_COMPILE_WITH_OSX_DEVICE_ - -#include "CIrrDeviceStub.h" -#include "IrrlichtDevice.h" -#include "IImagePresenter.h" -#include "IGUIEnvironment.h" -#include "ICursorControl.h" - -#include -#include - -class NSWindow; -class NSOpenGLContext; -class NSBitmapImageRep; - -namespace irr -{ - class CIrrDeviceMacOSX : public CIrrDeviceStub, video::IImagePresenter - { - public: - - //! constructor - CIrrDeviceMacOSX(const SIrrlichtCreationParameters& params); - - //! destructor - virtual ~CIrrDeviceMacOSX(); - - //! runs the device. Returns false if device wants to be deleted - virtual bool run(); - - //! Cause the device to temporarily pause execution and let other processes to run - // This should bring down processor usage without major performance loss for Irrlicht - virtual void yield(); - - //! Pause execution and let other processes to run for a specified amount of time. - virtual void sleep(u32 timeMs, bool pauseTimer); - - //! sets the caption of the window - virtual void setWindowCaption(const wchar_t* text); - - //! returns if window is active. if not, nothing need to be drawn - virtual bool isWindowActive() const; - - //! Checks if the Irrlicht window has focus - virtual bool isWindowFocused() const; - - //! Checks if the Irrlicht window is minimized - virtual bool isWindowMinimized() const; - - //! presents a surface in the client area - virtual bool present(video::IImage* surface, void* windowId=0, core::rect* src=0 ); - - //! notifies the device that it should close itself - virtual void closeDevice(); - - //! Sets if the window should be resizable in windowed mode. - virtual void setResizable(bool resize); - - //! Returns true if the window is resizable, false if not - virtual bool isResizable() const; - - //! Minimizes the window if possible - virtual void minimizeWindow(); - - //! Maximizes the window if possible. - virtual void maximizeWindow(); - - //! Restore the window to normal size if possible. - virtual void restoreWindow(); - - //! Activate any joysticks, and generate events for them. - virtual bool activateJoysticks(core::array & joystickInfo); - - //! \return Returns a pointer to a list with all video modes - //! supported by the gfx adapter. - virtual video::IVideoModeList* getVideoModeList(); - - //! Get the device type - virtual E_DEVICE_TYPE getType() const - { - return EIDT_OSX; - } - - void flush(); - void setMouseLocation(int x, int y); - void setResize(int width, int height); - void setCursorVisible(bool visible); - - private: - - //! create the driver - void createDriver(); - - //! Implementation of the macos x cursor control - class CCursorControl : public gui::ICursorControl - { - public: - - CCursorControl(const core::dimension2d& wsize, CIrrDeviceMacOSX *device) - : WindowSize(wsize), IsVisible(true), InvWindowSize(0.0f, 0.0f), Device(device), UseReferenceRect(false) - { - CursorPos.X = CursorPos.Y = 0; - if (WindowSize.Width!=0) - InvWindowSize.Width = 1.0f / WindowSize.Width; - if (WindowSize.Height!=0) - InvWindowSize.Height = 1.0f / WindowSize.Height; - } - - //! Changes the visible state of the mouse cursor. - virtual void setVisible(bool visible) - { - IsVisible = visible; - Device->setCursorVisible(visible); - } - - //! Returns if the cursor is currently visible. - virtual bool isVisible() const - { - return IsVisible; - } - - //! Sets the new position of the cursor. - virtual void setPosition(const core::position2d &pos) - { - setPosition(pos.X, pos.Y); - } - - //! Sets the new position of the cursor. - virtual void setPosition(f32 x, f32 y) - { - setPosition((s32)(x*WindowSize.Width), (s32)(y*WindowSize.Height)); - } - - //! Sets the new position of the cursor. - virtual void setPosition(const core::position2d &pos) - { - if (CursorPos.X != pos.X || CursorPos.Y != pos.Y) - setPosition(pos.X, pos.Y); - } - - //! Sets the new position of the cursor. - virtual void setPosition(s32 x, s32 y) - { - if (UseReferenceRect) - { - Device->setMouseLocation(ReferenceRect.UpperLeftCorner.X + x, ReferenceRect.UpperLeftCorner.Y + y); - } - else - { - Device->setMouseLocation(x,y); - } - } - - //! Returns the current position of the mouse cursor. - virtual const core::position2d& getPosition() - { - return CursorPos; - } - - //! Returns the current position of the mouse cursor. - virtual core::position2d getRelativePosition() - { - if (!UseReferenceRect) - { - return core::position2d(CursorPos.X * InvWindowSize.Width, - CursorPos.Y * InvWindowSize.Height); - } - - return core::position2d(CursorPos.X / (f32)ReferenceRect.getWidth(), - CursorPos.Y / (f32)ReferenceRect.getHeight()); - } - - //! Sets an absolute reference rect for calculating the cursor position. - virtual void setReferenceRect(core::rect* rect=0) - { - if (rect) - { - ReferenceRect = *rect; - UseReferenceRect = true; - - // prevent division through zero and uneven sizes - - if (!ReferenceRect.getHeight() || ReferenceRect.getHeight()%2) - ReferenceRect.LowerRightCorner.Y += 1; - - if (!ReferenceRect.getWidth() || ReferenceRect.getWidth()%2) - ReferenceRect.LowerRightCorner.X += 1; - } - else - UseReferenceRect = false; - } - - //! Updates the internal cursor position - void updateInternalCursorPosition(int x,int y) - { - CursorPos.X = x; - CursorPos.Y = y; - } - - private: - - core::position2d CursorPos; - core::dimension2d WindowSize; - core::dimension2d InvWindowSize; - core::rect ReferenceRect; - CIrrDeviceMacOSX *Device; - bool IsVisible; - bool UseReferenceRect; - }; - - bool createWindow(); - void initKeycodes(); - void storeMouseLocation(); - void postMouseEvent(void *event, irr::SEvent &ievent); - void postKeyEvent(void *event, irr::SEvent &ievent, bool pressed); - void pollJoysticks(); - - NSWindow *Window; - CGLContextObj CGLContext; - NSOpenGLContext *OGLContext; - NSBitmapImageRep *SoftwareDriverTarget; - std::map KeyCodes; - int DeviceWidth; - int DeviceHeight; - int ScreenWidth; - int ScreenHeight; - u32 MouseButtonStates; - u32 SoftwareRendererType; - bool IsFullscreen; - bool IsActive; - bool IsShiftDown; - bool IsControlDown; - bool IsResizable; - }; - - -} // end namespace irr - -#endif // _IRR_COMPILE_WITH_OSX_DEVICE_ -#endif // __C_IRR_DEVICE_MACOSX_H_INCLUDED__ - +// Copyright (C) 2005-2006 Etienne Petitjean +// Copyright (C) 2007-2012 Christian Stehno +// This file is part of the "Irrlicht Engine". +// For conditions of distribution and use, see copyright notice in Irrlicht.h + +#ifndef __C_IRR_DEVICE_MACOSX_H_INCLUDED__ +#define __C_IRR_DEVICE_MACOSX_H_INCLUDED__ + +#include "IrrCompileConfig.h" + +#ifdef _IRR_COMPILE_WITH_OSX_DEVICE_ + +#include "CIrrDeviceStub.h" +#include "IrrlichtDevice.h" +#include "IImagePresenter.h" +#include "IGUIEnvironment.h" +#include "ICursorControl.h" + +#include +#include + +class NSWindow; +class NSOpenGLContext; +class NSBitmapImageRep; + +namespace irr +{ + class CIrrDeviceMacOSX : public CIrrDeviceStub, video::IImagePresenter + { + public: + + //! constructor + CIrrDeviceMacOSX(const SIrrlichtCreationParameters& params); + + //! destructor + virtual ~CIrrDeviceMacOSX(); + + //! runs the device. Returns false if device wants to be deleted + virtual bool run(); + + //! Cause the device to temporarily pause execution and let other processes to run + // This should bring down processor usage without major performance loss for Irrlicht + virtual void yield(); + + //! Pause execution and let other processes to run for a specified amount of time. + virtual void sleep(u32 timeMs, bool pauseTimer); + + //! sets the caption of the window + virtual void setWindowCaption(const wchar_t* text); + + //! returns if window is active. if not, nothing need to be drawn + virtual bool isWindowActive() const; + + //! Checks if the Irrlicht window has focus + virtual bool isWindowFocused() const; + + //! Checks if the Irrlicht window is minimized + virtual bool isWindowMinimized() const; + + //! presents a surface in the client area + virtual bool present(video::IImage* surface, void* windowId=0, core::rect* src=0 ); + + //! notifies the device that it should close itself + virtual void closeDevice(); + + //! Sets if the window should be resizable in windowed mode. + virtual void setResizable(bool resize); + + //! Returns true if the window is resizable, false if not + virtual bool isResizable() const; + + //! Minimizes the window if possible + virtual void minimizeWindow(); + + //! Maximizes the window if possible. + virtual void maximizeWindow(); + + //! Restore the window to normal size if possible. + virtual void restoreWindow(); + + //! Activate any joysticks, and generate events for them. + virtual bool activateJoysticks(core::array & joystickInfo); + + //! \return Returns a pointer to a list with all video modes + //! supported by the gfx adapter. + virtual video::IVideoModeList* getVideoModeList(); + + //! Get the device type + virtual E_DEVICE_TYPE getType() const + { + return EIDT_OSX; + } + + void flush(); + void setMouseLocation(int x, int y); + void setResize(int width, int height); + void setCursorVisible(bool visible); + + private: + + //! create the driver + void createDriver(); + + //! Implementation of the macos x cursor control + class CCursorControl : public gui::ICursorControl + { + public: + + CCursorControl(const core::dimension2d& wsize, CIrrDeviceMacOSX *device) + : WindowSize(wsize), IsVisible(true), InvWindowSize(0.0f, 0.0f), Device(device), UseReferenceRect(false) + { + CursorPos.X = CursorPos.Y = 0; + if (WindowSize.Width!=0) + InvWindowSize.Width = 1.0f / WindowSize.Width; + if (WindowSize.Height!=0) + InvWindowSize.Height = 1.0f / WindowSize.Height; + } + + //! Changes the visible state of the mouse cursor. + virtual void setVisible(bool visible) + { + IsVisible = visible; + Device->setCursorVisible(visible); + } + + //! Returns if the cursor is currently visible. + virtual bool isVisible() const + { + return IsVisible; + } + + //! Sets the new position of the cursor. + virtual void setPosition(const core::position2d &pos) + { + setPosition(pos.X, pos.Y); + } + + //! Sets the new position of the cursor. + virtual void setPosition(f32 x, f32 y) + { + setPosition((s32)(x*WindowSize.Width), (s32)(y*WindowSize.Height)); + } + + //! Sets the new position of the cursor. + virtual void setPosition(const core::position2d &pos) + { + if (CursorPos.X != pos.X || CursorPos.Y != pos.Y) + setPosition(pos.X, pos.Y); + } + + //! Sets the new position of the cursor. + virtual void setPosition(s32 x, s32 y) + { + if (UseReferenceRect) + { + Device->setMouseLocation(ReferenceRect.UpperLeftCorner.X + x, ReferenceRect.UpperLeftCorner.Y + y); + } + else + { + Device->setMouseLocation(x,y); + } + } + + //! Returns the current position of the mouse cursor. + virtual const core::position2d& getPosition() + { + return CursorPos; + } + + //! Returns the current position of the mouse cursor. + virtual core::position2d getRelativePosition() + { + if (!UseReferenceRect) + { + return core::position2d(CursorPos.X * InvWindowSize.Width, + CursorPos.Y * InvWindowSize.Height); + } + + return core::position2d(CursorPos.X / (f32)ReferenceRect.getWidth(), + CursorPos.Y / (f32)ReferenceRect.getHeight()); + } + + //! Sets an absolute reference rect for calculating the cursor position. + virtual void setReferenceRect(core::rect* rect=0) + { + if (rect) + { + ReferenceRect = *rect; + UseReferenceRect = true; + + // prevent division through zero and uneven sizes + + if (!ReferenceRect.getHeight() || ReferenceRect.getHeight()%2) + ReferenceRect.LowerRightCorner.Y += 1; + + if (!ReferenceRect.getWidth() || ReferenceRect.getWidth()%2) + ReferenceRect.LowerRightCorner.X += 1; + } + else + UseReferenceRect = false; + } + + //! Updates the internal cursor position + void updateInternalCursorPosition(int x,int y) + { + CursorPos.X = x; + CursorPos.Y = y; + } + + private: + + core::position2d CursorPos; + core::dimension2d WindowSize; + core::dimension2d InvWindowSize; + core::rect ReferenceRect; + CIrrDeviceMacOSX *Device; + bool IsVisible; + bool UseReferenceRect; + }; + + bool createWindow(); + void initKeycodes(); + void storeMouseLocation(); + void postMouseEvent(void *event, irr::SEvent &ievent); + void postKeyEvent(void *event, irr::SEvent &ievent, bool pressed); + void pollJoysticks(); + + NSWindow *Window; + CGLContextObj CGLContext; + NSOpenGLContext *OGLContext; + NSBitmapImageRep *SoftwareDriverTarget; + std::map KeyCodes; + int DeviceWidth; + int DeviceHeight; + int ScreenWidth; + int ScreenHeight; + u32 MouseButtonStates; + u32 SoftwareRendererType; + bool IsFullscreen; + bool IsActive; + bool IsShiftDown; + bool IsControlDown; + bool IsResizable; + }; + + +} // end namespace irr + +#endif // _IRR_COMPILE_WITH_OSX_DEVICE_ +#endif // __C_IRR_DEVICE_MACOSX_H_INCLUDED__ + diff --git a/libraries/irrlicht-1.8/source/Irrlicht/MacOSX/CIrrDeviceMacOSX.mm b/libraries/irrlicht-1.8/source/Irrlicht/MacOSX/CIrrDeviceMacOSX.mm index 43453f4..6862c0a 100644 --- a/libraries/irrlicht-1.8/source/Irrlicht/MacOSX/CIrrDeviceMacOSX.mm +++ b/libraries/irrlicht-1.8/source/Irrlicht/MacOSX/CIrrDeviceMacOSX.mm @@ -1,1892 +1,1892 @@ -// Copyright (C) 2005-2006 Etienne Petitjean -// Copyright (C) 2007-2012 Christian Stehno -// This file is part of the "Irrlicht Engine". -// For conditions of distribution and use, see copyright notice in Irrlicht.h - -#include "IrrCompileConfig.h" - -#ifdef _IRR_COMPILE_WITH_OSX_DEVICE_ - -#import -#import -#ifndef __MAC_10_6 -#import -#endif - -#include "CIrrDeviceMacOSX.h" -#include "IEventReceiver.h" -#include "irrList.h" -#include "os.h" -#include "CTimer.h" -#include "irrString.h" -#include "Keycodes.h" -#include -#include -#include "COSOperator.h" -#include "CColorConverter.h" -#include "irrlicht.h" - - -#import -#import -#import "AppDelegate.h" - -#if defined _IRR_COMPILE_WITH_JOYSTICK_EVENTS_ - -#include -#include -#ifdef MACOS_10_0_4 -#include -#else -/* The header was moved here in Mac OS X 10.1 */ -#include -#endif -#include -#include - -// only OSX 10.5 seems to not need these defines... -#if !defined(__MAC_10_5) || defined(__MAC_10_6) -// Contents from Events.h from Carbon/HIToolbox but we need it with Cocoa too -// and for some reason no Cocoa equivalent of these constants seems provided. -// So I'm doing like everyone else and using copy-and-paste. - -/* - * Summary: - * Virtual keycodes - * - * Discussion: - * These constants are the virtual keycodes defined originally in - * Inside Mac Volume V, pg. V-191. They identify physical keys on a - * keyboard. Those constants with "ANSI" in the name are labeled - * according to the key position on an ANSI-standard US keyboard. - * For example, kVK_ANSI_A indicates the virtual keycode for the key - * with the letter 'A' in the US keyboard layout. Other keyboard - * layouts may have the 'A' key label on a different physical key; - * in this case, pressing 'A' will generate a different virtual - * keycode. - */ -enum { - kVK_ANSI_A = 0x00, - kVK_ANSI_S = 0x01, - kVK_ANSI_D = 0x02, - kVK_ANSI_F = 0x03, - kVK_ANSI_H = 0x04, - kVK_ANSI_G = 0x05, - kVK_ANSI_Z = 0x06, - kVK_ANSI_X = 0x07, - kVK_ANSI_C = 0x08, - kVK_ANSI_V = 0x09, - kVK_ANSI_B = 0x0B, - kVK_ANSI_Q = 0x0C, - kVK_ANSI_W = 0x0D, - kVK_ANSI_E = 0x0E, - kVK_ANSI_R = 0x0F, - kVK_ANSI_Y = 0x10, - kVK_ANSI_T = 0x11, - kVK_ANSI_1 = 0x12, - kVK_ANSI_2 = 0x13, - kVK_ANSI_3 = 0x14, - kVK_ANSI_4 = 0x15, - kVK_ANSI_6 = 0x16, - kVK_ANSI_5 = 0x17, - kVK_ANSI_Equal = 0x18, - kVK_ANSI_9 = 0x19, - kVK_ANSI_7 = 0x1A, - kVK_ANSI_Minus = 0x1B, - kVK_ANSI_8 = 0x1C, - kVK_ANSI_0 = 0x1D, - kVK_ANSI_RightBracket = 0x1E, - kVK_ANSI_O = 0x1F, - kVK_ANSI_U = 0x20, - kVK_ANSI_LeftBracket = 0x21, - kVK_ANSI_I = 0x22, - kVK_ANSI_P = 0x23, - kVK_ANSI_L = 0x25, - kVK_ANSI_J = 0x26, - kVK_ANSI_Quote = 0x27, - kVK_ANSI_K = 0x28, - kVK_ANSI_Semicolon = 0x29, - kVK_ANSI_Backslash = 0x2A, - kVK_ANSI_Comma = 0x2B, - kVK_ANSI_Slash = 0x2C, - kVK_ANSI_N = 0x2D, - kVK_ANSI_M = 0x2E, - kVK_ANSI_Period = 0x2F, - kVK_ANSI_Grave = 0x32, - kVK_ANSI_KeypadDecimal = 0x41, - kVK_ANSI_KeypadMultiply = 0x43, - kVK_ANSI_KeypadPlus = 0x45, - kVK_ANSI_KeypadClear = 0x47, - kVK_ANSI_KeypadDivide = 0x4B, - kVK_ANSI_KeypadEnter = 0x4C, - kVK_ANSI_KeypadMinus = 0x4E, - kVK_ANSI_KeypadEquals = 0x51, - kVK_ANSI_Keypad0 = 0x52, - kVK_ANSI_Keypad1 = 0x53, - kVK_ANSI_Keypad2 = 0x54, - kVK_ANSI_Keypad3 = 0x55, - kVK_ANSI_Keypad4 = 0x56, - kVK_ANSI_Keypad5 = 0x57, - kVK_ANSI_Keypad6 = 0x58, - kVK_ANSI_Keypad7 = 0x59, - kVK_ANSI_Keypad8 = 0x5B, - kVK_ANSI_Keypad9 = 0x5C -}; - -/* keycodes for keys that are independent of keyboard layout*/ -enum { - kVK_Return = 0x24, - kVK_Tab = 0x30, - kVK_Space = 0x31, - kVK_Delete = 0x33, - kVK_Escape = 0x35, - kVK_Command = 0x37, - kVK_Shift = 0x38, - kVK_CapsLock = 0x39, - kVK_Option = 0x3A, - kVK_Control = 0x3B, - kVK_RightShift = 0x3C, - kVK_RightOption = 0x3D, - kVK_RightControl = 0x3E, - kVK_Function = 0x3F, - kVK_F17 = 0x40, - kVK_VolumeUp = 0x48, - kVK_VolumeDown = 0x49, - kVK_Mute = 0x4A, - kVK_F18 = 0x4F, - kVK_F19 = 0x50, - kVK_F20 = 0x5A, - kVK_F5 = 0x60, - kVK_F6 = 0x61, - kVK_F7 = 0x62, - kVK_F3 = 0x63, - kVK_F8 = 0x64, - kVK_F9 = 0x65, - kVK_F11 = 0x67, - kVK_F13 = 0x69, - kVK_F16 = 0x6A, - kVK_F14 = 0x6B, - kVK_F10 = 0x6D, - kVK_F12 = 0x6F, - kVK_F15 = 0x71, - kVK_Help = 0x72, - kVK_Home = 0x73, - kVK_PageUp = 0x74, - kVK_ForwardDelete = 0x75, - kVK_F4 = 0x76, - kVK_End = 0x77, - kVK_F2 = 0x78, - kVK_PageDown = 0x79, - kVK_F1 = 0x7A, - kVK_LeftArrow = 0x7B, - kVK_RightArrow = 0x7C, - kVK_DownArrow = 0x7D, - kVK_UpArrow = 0x7E -}; -#endif - -struct JoystickComponent -{ - IOHIDElementCookie cookie; // unique value which identifies element, will NOT change - long min; // reported min value possible - long max; // reported max value possible - - long minRead; //min read value - long maxRead; //max read value - - JoystickComponent() : min(0), minRead(0), max(0), maxRead(0) - { - } -}; - -struct JoystickInfo -{ - irr::core::array axisComp; - irr::core::array buttonComp; - irr::core::array hatComp; - - int hats; - int axes; - int buttons; - int numActiveJoysticks; - - irr::SEvent persistentData; - - IOHIDDeviceInterface ** interface; - bool removed; - char joystickName[256]; - long usage; // usage page from IOUSBHID Parser.h which defines general usage - long usagePage; // usage within above page from IOUSBHID Parser.h which defines specific usage - - JoystickInfo() : hats(0), axes(0), buttons(0), interface(0), removed(false), usage(0), usagePage(0), numActiveJoysticks(0) - { - interface = NULL; - memset(joystickName, '\0', 256); - axisComp.clear(); - buttonComp.clear(); - hatComp.clear(); - - persistentData.EventType = irr::EET_JOYSTICK_INPUT_EVENT; - persistentData.JoystickEvent.POV = 65535; - persistentData.JoystickEvent.ButtonStates = 0; - } -}; -irr::core::array ActiveJoysticks; - -//helper functions for init joystick -static IOReturn closeJoystickDevice (JoystickInfo* joyInfo) -{ - IOReturn result = kIOReturnSuccess; - if (joyInfo && joyInfo->interface) - { - /* close the interface */ - result = (*(joyInfo->interface))->close (joyInfo->interface); - if (kIOReturnNotOpen == result) - { - /* do nothing as device was not opened, thus can't be closed */ - } - else if (kIOReturnSuccess != result) - irr::os::Printer::log("IOHIDDeviceInterface failed to close", irr::ELL_ERROR); - /* release the interface */ - result = (*(joyInfo->interface))->Release (joyInfo->interface); - if (kIOReturnSuccess != result) - irr::os::Printer::log("IOHIDDeviceInterface failed to release", irr::ELL_ERROR); - joyInfo->interface = NULL; - } - return result; -} - -static void addComponentInfo (CFTypeRef refElement, JoystickComponent *pComponent, int numActiveJoysticks) -{ - long number; - CFTypeRef refType; - - refType = CFDictionaryGetValue ((CFDictionaryRef)refElement, CFSTR(kIOHIDElementCookieKey)); - if (refType && CFNumberGetValue ((CFNumberRef)refType, kCFNumberLongType, &number)) - pComponent->cookie = (IOHIDElementCookie) number; - refType = CFDictionaryGetValue ((CFDictionaryRef)refElement, CFSTR(kIOHIDElementMinKey)); - if (refType && CFNumberGetValue ((CFNumberRef)refType, kCFNumberLongType, &number)) - pComponent->minRead = pComponent->min = number; - refType = CFDictionaryGetValue ((CFDictionaryRef)refElement, CFSTR(kIOHIDElementMaxKey)); - if (refType && CFNumberGetValue ((CFNumberRef)refType, kCFNumberLongType, &number)) - pComponent->maxRead = pComponent->max = number; -} - -static void getJoystickComponentArrayHandler (const void * value, void * parameter); - -static void addJoystickComponent (CFTypeRef refElement, JoystickInfo* joyInfo) -{ - long elementType, usagePage, usage; - CFTypeRef refElementType = CFDictionaryGetValue ((CFDictionaryRef)refElement, CFSTR(kIOHIDElementTypeKey)); - CFTypeRef refUsagePage = CFDictionaryGetValue ((CFDictionaryRef)refElement, CFSTR(kIOHIDElementUsagePageKey)); - CFTypeRef refUsage = CFDictionaryGetValue ((CFDictionaryRef)refElement, CFSTR(kIOHIDElementUsageKey)); - - if ((refElementType) && (CFNumberGetValue ((CFNumberRef)refElementType, kCFNumberLongType, &elementType))) - { - /* look at types of interest */ - if ((elementType == kIOHIDElementTypeInput_Misc) || (elementType == kIOHIDElementTypeInput_Button) || - (elementType == kIOHIDElementTypeInput_Axis)) - { - if (refUsagePage && CFNumberGetValue ((CFNumberRef)refUsagePage, kCFNumberLongType, &usagePage) && - refUsage && CFNumberGetValue ((CFNumberRef)refUsage, kCFNumberLongType, &usage)) - { - switch (usagePage) /* only interested in kHIDPage_GenericDesktop and kHIDPage_Button */ - { - case kHIDPage_GenericDesktop: - { - switch (usage) /* look at usage to determine function */ - { - case kHIDUsage_GD_X: - case kHIDUsage_GD_Y: - case kHIDUsage_GD_Z: - case kHIDUsage_GD_Rx: - case kHIDUsage_GD_Ry: - case kHIDUsage_GD_Rz: - case kHIDUsage_GD_Slider: - case kHIDUsage_GD_Dial: - case kHIDUsage_GD_Wheel: - { - joyInfo->axes++; - JoystickComponent newComponent; - addComponentInfo(refElement, &newComponent, joyInfo->numActiveJoysticks); - joyInfo->axisComp.push_back(newComponent); - } - break; - case kHIDUsage_GD_Hatswitch: - { - joyInfo->hats++; - JoystickComponent newComponent; - addComponentInfo(refElement, &newComponent, joyInfo->numActiveJoysticks); - joyInfo->hatComp.push_back(newComponent); - } - break; - } - } - break; - case kHIDPage_Button: - { - joyInfo->buttons++; - JoystickComponent newComponent; - addComponentInfo(refElement, &newComponent, joyInfo->numActiveJoysticks); - joyInfo->buttonComp.push_back(newComponent); - } - break; - default: - break; - } - } - } - else if (kIOHIDElementTypeCollection == elementType) - { - //get elements - CFTypeRef refElementTop = CFDictionaryGetValue ((CFMutableDictionaryRef) refElement, CFSTR(kIOHIDElementKey)); - if (refElementTop) - { - CFTypeID type = CFGetTypeID (refElementTop); - if (type == CFArrayGetTypeID()) - { - CFRange range = {0, CFArrayGetCount ((CFArrayRef)refElementTop)}; - CFArrayApplyFunction ((CFArrayRef)refElementTop, range, getJoystickComponentArrayHandler, joyInfo); - } - } - } - } -} - -static void getJoystickComponentArrayHandler (const void * value, void * parameter) -{ - if (CFGetTypeID (value) == CFDictionaryGetTypeID ()) - addJoystickComponent ((CFTypeRef) value, (JoystickInfo *) parameter); -} - -static void joystickTopLevelElementHandler (const void * value, void * parameter) -{ - CFTypeRef refCF = 0; - if (CFGetTypeID (value) != CFDictionaryGetTypeID ()) - return; - refCF = CFDictionaryGetValue ((CFDictionaryRef)value, CFSTR(kIOHIDElementUsagePageKey)); - if (!CFNumberGetValue ((CFNumberRef)refCF, kCFNumberLongType, &((JoystickInfo *) parameter)->usagePage)) - irr::os::Printer::log("CFNumberGetValue error retrieving JoystickInfo->usagePage", irr::ELL_ERROR); - refCF = CFDictionaryGetValue ((CFDictionaryRef)value, CFSTR(kIOHIDElementUsageKey)); - if (!CFNumberGetValue ((CFNumberRef)refCF, kCFNumberLongType, &((JoystickInfo *) parameter)->usage)) - irr::os::Printer::log("CFNumberGetValue error retrieving JoystickInfo->usage", irr::ELL_ERROR); -} - -static void getJoystickDeviceInfo (io_object_t hidDevice, CFMutableDictionaryRef hidProperties, JoystickInfo *joyInfo) -{ - CFMutableDictionaryRef usbProperties = 0; - io_registry_entry_t parent1, parent2; - - /* Mac OS X currently is not mirroring all USB properties to HID page so need to look at USB device page also - * get dictionary for usb properties: step up two levels and get CF dictionary for USB properties - */ - if ((KERN_SUCCESS == IORegistryEntryGetParentEntry (hidDevice, kIOServicePlane, &parent1)) && - (KERN_SUCCESS == IORegistryEntryGetParentEntry (parent1, kIOServicePlane, &parent2)) && - (KERN_SUCCESS == IORegistryEntryCreateCFProperties (parent2, &usbProperties, kCFAllocatorDefault, kNilOptions))) - { - if (usbProperties) - { - CFTypeRef refCF = 0; - /* get device info - * try hid dictionary first, if fail then go to usb dictionary - */ - - /* get joystickName name */ - refCF = CFDictionaryGetValue (hidProperties, CFSTR(kIOHIDProductKey)); - if (!refCF) - refCF = CFDictionaryGetValue (usbProperties, CFSTR("USB Product Name")); - if (refCF) - { - if (!CFStringGetCString ((CFStringRef)refCF, joyInfo->joystickName, 256, CFStringGetSystemEncoding ())) - irr::os::Printer::log("CFStringGetCString error getting joyInfo->joystickName", irr::ELL_ERROR); - } - - /* get usage page and usage */ - refCF = CFDictionaryGetValue (hidProperties, CFSTR(kIOHIDPrimaryUsagePageKey)); - if (refCF) - { - if (!CFNumberGetValue ((CFNumberRef)refCF, kCFNumberLongType, &joyInfo->usagePage)) - irr::os::Printer::log("CFNumberGetValue error getting joyInfo->usagePage", irr::ELL_ERROR); - refCF = CFDictionaryGetValue (hidProperties, CFSTR(kIOHIDPrimaryUsageKey)); - if (refCF) - if (!CFNumberGetValue ((CFNumberRef)refCF, kCFNumberLongType, &joyInfo->usage)) - irr::os::Printer::log("CFNumberGetValue error getting joyInfo->usage", irr::ELL_ERROR); - } - - if (NULL == refCF) /* get top level element HID usage page or usage */ - { - /* use top level element instead */ - CFTypeRef refCFTopElement = 0; - refCFTopElement = CFDictionaryGetValue (hidProperties, CFSTR(kIOHIDElementKey)); - { - /* refCFTopElement points to an array of element dictionaries */ - CFRange range = {0, CFArrayGetCount ((CFArrayRef)refCFTopElement)}; - CFArrayApplyFunction ((CFArrayRef)refCFTopElement, range, joystickTopLevelElementHandler, joyInfo); - } - } - - CFRelease (usbProperties); - } - else - irr::os::Printer::log("IORegistryEntryCreateCFProperties failed to create usbProperties", irr::ELL_ERROR); - - if (kIOReturnSuccess != IOObjectRelease (parent2)) - irr::os::Printer::log("IOObjectRelease failed to release parent2", irr::ELL_ERROR); - if (kIOReturnSuccess != IOObjectRelease (parent1)) - irr::os::Printer::log("IOObjectRelease failed to release parent1", irr::ELL_ERROR); - } -} - -#endif // _IRR_COMPILE_WITH_JOYSTICK_EVENTS_ - -//------------------------------------------------------------------------------------------ -Boolean GetDictionaryBoolean(CFDictionaryRef theDict, const void* key) -{ - // get a boolean from the dictionary - Boolean value = false; - CFBooleanRef boolRef; - boolRef = (CFBooleanRef)CFDictionaryGetValue(theDict, key); - if (boolRef != NULL) - value = CFBooleanGetValue(boolRef); - return value; -} -//------------------------------------------------------------------------------------------ -long GetDictionaryLong(CFDictionaryRef theDict, const void* key) -{ - // get a long from the dictionary - long value = 0; - CFNumberRef numRef; - numRef = (CFNumberRef)CFDictionaryGetValue(theDict, key); - if (numRef != NULL) - CFNumberGetValue(numRef, kCFNumberLongType, &value); - return value; -} - -namespace irr -{ - namespace video - { - IVideoDriver* createOpenGLDriver(const SIrrlichtCreationParameters& param, io::IFileSystem* io, CIrrDeviceMacOSX *device); - } -} // end namespace irr - -static bool firstLaunch = true; - -namespace irr -{ -//! constructor -CIrrDeviceMacOSX::CIrrDeviceMacOSX(const SIrrlichtCreationParameters& param) - : CIrrDeviceStub(param), Window(NULL), CGLContext(NULL), OGLContext(NULL), - SoftwareDriverTarget(0), DeviceWidth(0), DeviceHeight(0), - ScreenWidth(0), ScreenHeight(0), MouseButtonStates(0), SoftwareRendererType(0), - IsActive(true), IsFullscreen(false), IsShiftDown(false), IsControlDown(false), IsResizable(false) -{ - struct utsname name; - NSString *path; - - #ifdef _DEBUG - setDebugName("CIrrDeviceMacOSX"); - #endif - - if (firstLaunch) - { - firstLaunch = false; - - if(!CreationParams.WindowId) //load menus if standalone application - { - [[NSAutoreleasePool alloc] init]; - [NSApplication sharedApplication]; - [NSApp setDelegate:[[[AppDelegate alloc] initWithDevice:this] autorelease]]; - [NSBundle loadNibNamed:@"MainMenu" owner:[NSApp delegate]]; - [NSApp finishLaunching]; - } - - path = [[[NSBundle mainBundle] bundlePath] stringByDeletingLastPathComponent]; - chdir([path fileSystemRepresentation]); - [path release]; - } - - uname(&name); - Operator = new COSOperator(name.version); - os::Printer::log(name.version,ELL_INFORMATION); - - initKeycodes(); - - VideoModeList.setDesktop(CreationParams.Bits, core::dimension2d([[NSScreen mainScreen] frame].size.width, [[NSScreen mainScreen] frame].size.height)); - - bool success = true; - if (CreationParams.DriverType != video::EDT_NULL) - success = createWindow(); - - // in case of failure, one can check VideoDriver for initialization - if (!success) - return; - - setResizable(false); - CursorControl = new CCursorControl(CreationParams.WindowSize, this); - - createDriver(); - createGUIAndScene(); -} - -CIrrDeviceMacOSX::~CIrrDeviceMacOSX() -{ - [SoftwareDriverTarget release]; -#ifdef __MAC_10_6 - [NSApp setPresentationOptions:(NSApplicationPresentationDefault)]; -#else - SetSystemUIMode(kUIModeNormal, kUIOptionAutoShowMenuBar); -#endif - closeDevice(); -#if defined(_IRR_COMPILE_WITH_JOYSTICK_EVENTS_) - for (u32 joystick = 0; joystick < ActiveJoysticks.size(); ++joystick) - { - if (ActiveJoysticks[joystick].interface) - closeJoystickDevice(&ActiveJoysticks[joystick]); - } -#endif -} - -void CIrrDeviceMacOSX::closeDevice() -{ - if (Window != NULL) - { - [Window setIsVisible:FALSE]; - - if (OGLContext != NULL) - { - [OGLContext clearDrawable]; - [OGLContext release]; - OGLContext = NULL; - } - - [Window setReleasedWhenClosed:TRUE]; - [Window release]; - Window = NULL; - - if (IsFullscreen) - CGReleaseAllDisplays(); - } - else - { - if (CGLContext != NULL) - { - if(CreationParams.WindowId) - { - [(NSOpenGLContext *)OGLContext clearDrawable]; - [(NSOpenGLContext *)OGLContext release]; - OGLContext = NULL; - } - else - { - CGLSetCurrentContext(NULL); - CGLClearDrawable(CGLContext); - CGLDestroyContext(CGLContext); - CGReleaseAllDisplays(); - } - } - } - - IsFullscreen = false; - IsActive = false; - CGLContext = NULL; -} - -bool CIrrDeviceMacOSX::createWindow() -{ - CGDisplayErr error; - bool result=false; - CGDirectDisplayID display=CGMainDisplayID(); - CGLPixelFormatObj pixelFormat; - CGRect displayRect; -#ifdef __MAC_10_6 - CGDisplayModeRef displaymode, olddisplaymode; -#else - CFDictionaryRef displaymode, olddisplaymode; -#endif - GLint numPixelFormats, newSwapInterval; - - int alphaSize = CreationParams.WithAlphaChannel?4:0; - int depthSize = CreationParams.ZBufferBits; - if (CreationParams.WithAlphaChannel && (CreationParams.Bits == 32)) - alphaSize = 8; - - ScreenWidth = (int) CGDisplayPixelsWide(display); - ScreenHeight = (int) CGDisplayPixelsHigh(display); - - // we need to check where the exceptions may happen and work at them - // for now we will just catch them to be able to avoid an app exit - @try - { - if (!CreationParams.Fullscreen) - { - if(!CreationParams.WindowId) //create another window when WindowId is null - { - NSBackingStoreType type = (CreationParams.DriverType == video::EDT_OPENGL) ? NSBackingStoreBuffered : NSBackingStoreNonretained; - - Window = [[NSWindow alloc] initWithContentRect:NSMakeRect(0,0,CreationParams.WindowSize.Width,CreationParams.WindowSize.Height) styleMask:NSTitledWindowMask+NSClosableWindowMask+NSResizableWindowMask backing:type defer:FALSE]; - } - - if (Window != NULL || CreationParams.WindowId) - { - if (CreationParams.DriverType == video::EDT_OPENGL) - { - NSOpenGLPixelFormatAttribute windowattribs[] = - { - NSOpenGLPFANoRecovery, - NSOpenGLPFAAccelerated, - NSOpenGLPFADepthSize, (NSOpenGLPixelFormatAttribute)depthSize, - NSOpenGLPFAColorSize, (NSOpenGLPixelFormatAttribute)CreationParams.Bits, - NSOpenGLPFAAlphaSize, (NSOpenGLPixelFormatAttribute)alphaSize, - NSOpenGLPFASampleBuffers, (NSOpenGLPixelFormatAttribute)1, - NSOpenGLPFASamples, (NSOpenGLPixelFormatAttribute)CreationParams.AntiAlias, - NSOpenGLPFAStencilSize, (NSOpenGLPixelFormatAttribute)(CreationParams.Stencilbuffer?1:0), - NSOpenGLPFADoubleBuffer, - (NSOpenGLPixelFormatAttribute)nil - }; - - if (CreationParams.AntiAlias<2) - { - windowattribs[ 9] = (NSOpenGLPixelFormatAttribute)0; - windowattribs[11] = (NSOpenGLPixelFormatAttribute)0; - } - - NSOpenGLPixelFormat *format; - for (int i=0; i<3; ++i) - { - if (1==i) - { - // Second try without stencilbuffer - if (CreationParams.Stencilbuffer) - { - windowattribs[13]=(NSOpenGLPixelFormatAttribute)0; - } - else - continue; - } - else if (2==i) - { - // Third try without Doublebuffer - os::Printer::log("No doublebuffering available.", ELL_WARNING); - windowattribs[14]=(NSOpenGLPixelFormatAttribute)nil; - } - - format = [[NSOpenGLPixelFormat alloc] initWithAttributes:windowattribs]; - if (format == NULL) - { - if (CreationParams.AntiAlias>1) - { - while (!format && windowattribs[12]>1) - { - windowattribs[12] = (NSOpenGLPixelFormatAttribute)((int)windowattribs[12]-1); - format = [[NSOpenGLPixelFormat alloc] initWithAttributes:windowattribs]; - } - - if (!format) - { - windowattribs[9] = (NSOpenGLPixelFormatAttribute)0; - windowattribs[11] = (NSOpenGLPixelFormatAttribute)0; - format = [[NSOpenGLPixelFormat alloc] initWithAttributes:windowattribs]; - if (!format) - { - // reset values for next try - windowattribs[9] = (NSOpenGLPixelFormatAttribute)1; - windowattribs[11] = (NSOpenGLPixelFormatAttribute)CreationParams.AntiAlias; - } - else - { - os::Printer::log("No FSAA available.", ELL_WARNING); - } - } - } - } - else - break; - } - CreationParams.AntiAlias = windowattribs[11]; - CreationParams.Stencilbuffer=(windowattribs[13]==1); - - if (format != NULL) - { - OGLContext = [[NSOpenGLContext alloc] initWithFormat:format shareContext:NULL]; - [format release]; - } - } - - if (OGLContext != NULL || CreationParams.DriverType != video::EDT_OPENGL) - { - if (!CreationParams.WindowId) - { - [Window center]; - [Window setDelegate:[NSApp delegate]]; - - if(CreationParams.DriverType == video::EDT_OPENGL) - [OGLContext setView:[Window contentView]]; - - [Window setAcceptsMouseMovedEvents:TRUE]; - [Window setIsVisible:TRUE]; - [Window makeKeyAndOrderFront:nil]; - } - else if(CreationParams.DriverType == video::EDT_OPENGL) //use another window for drawing - [OGLContext setView:(NSView*)CreationParams.WindowId]; - - if (CreationParams.DriverType == video::EDT_OPENGL) - CGLContext = (CGLContextObj) [OGLContext CGLContextObj]; - - DeviceWidth = CreationParams.WindowSize.Width; - DeviceHeight = CreationParams.WindowSize.Height; - result = true; - } - } - } - else - { - IsFullscreen = true; - -#ifdef __MAC_10_6 - displaymode = CGDisplayCopyDisplayMode(display); - - CFArrayRef Modes = CGDisplayCopyAllDisplayModes(display, NULL); - - for(int i = 0; i < CFArrayGetCount(Modes); ++i) - { - CGDisplayModeRef CurrentMode = (CGDisplayModeRef)CFArrayGetValueAtIndex(Modes, i); - - u8 Depth = 0; - - CFStringRef pixEnc = CGDisplayModeCopyPixelEncoding(CurrentMode); - - if(CFStringCompare(pixEnc, CFSTR(IO32BitDirectPixels), kCFCompareCaseInsensitive) == kCFCompareEqualTo) - Depth = 32; - else - if(CFStringCompare(pixEnc, CFSTR(IO16BitDirectPixels), kCFCompareCaseInsensitive) == kCFCompareEqualTo) - Depth = 16; - else - if(CFStringCompare(pixEnc, CFSTR(IO8BitIndexedPixels), kCFCompareCaseInsensitive) == kCFCompareEqualTo) - Depth = 8; - - if(Depth == CreationParams.Bits) - if((CGDisplayModeGetWidth(CurrentMode) == CreationParams.WindowSize.Width) && (CGDisplayModeGetHeight(CurrentMode) == CreationParams.WindowSize.Height)) - { - displaymode = CurrentMode; - break; - } - } -#else - displaymode = CGDisplayBestModeForParameters(display,CreationParams.Bits,CreationParams.WindowSize.Width,CreationParams.WindowSize.Height,NULL); -#endif - - if (displaymode != NULL) - { -#ifdef __MAC_10_6 - olddisplaymode = CGDisplayCopyDisplayMode(display); -#else - olddisplaymode = CGDisplayCurrentMode(display); -#endif - - error = CGCaptureAllDisplays(); - if (error == CGDisplayNoErr) - { -#ifdef __MAC_10_6 - error = CGDisplaySetDisplayMode(display, displaymode, NULL); -#else - error = CGDisplaySwitchToMode(display, displaymode); -#endif - - if (error == CGDisplayNoErr) - { - if (CreationParams.DriverType == video::EDT_OPENGL) - { - CGLPixelFormatAttribute fullattribs[] = - { - kCGLPFAFullScreen, - kCGLPFADisplayMask, (CGLPixelFormatAttribute)CGDisplayIDToOpenGLDisplayMask(display), - kCGLPFADoubleBuffer, - kCGLPFANoRecovery, - kCGLPFAAccelerated, - kCGLPFADepthSize, (CGLPixelFormatAttribute)depthSize, - kCGLPFAColorSize, (CGLPixelFormatAttribute)CreationParams.Bits, - kCGLPFAAlphaSize, (CGLPixelFormatAttribute)alphaSize, - kCGLPFASampleBuffers, (CGLPixelFormatAttribute)(CreationParams.AntiAlias?1:0), - kCGLPFASamples, (CGLPixelFormatAttribute)CreationParams.AntiAlias, - kCGLPFAStencilSize, (CGLPixelFormatAttribute)(CreationParams.Stencilbuffer?1:0), - (CGLPixelFormatAttribute)NULL - }; - - pixelFormat = NULL; - numPixelFormats = 0; - CGLChoosePixelFormat(fullattribs,&pixelFormat,&numPixelFormats); - - if (pixelFormat != NULL) - { - CGLCreateContext(pixelFormat,NULL,&CGLContext); - CGLDestroyPixelFormat(pixelFormat); - } - - if (CGLContext != NULL) - { -#ifdef __MAC_10_6 - CGLSetFullScreenOnDisplay(CGLContext, CGDisplayIDToOpenGLDisplayMask(display)); -#else - CGLSetFullScreen(CGLContext); -#endif - displayRect = CGDisplayBounds(display); - ScreenWidth = DeviceWidth = (int)displayRect.size.width; - ScreenHeight = DeviceHeight = (int)displayRect.size.height; - CreationParams.WindowSize.set(ScreenWidth, ScreenHeight); - result = true; - } - } - else - { - Window = [[NSWindow alloc] initWithContentRect:[[NSScreen mainScreen] frame] styleMask:NSBorderlessWindowMask backing:NSBackingStoreNonretained defer:NO screen:[NSScreen mainScreen]]; - - [Window setLevel: CGShieldingWindowLevel()]; - [Window setAcceptsMouseMovedEvents:TRUE]; - [Window setIsVisible:TRUE]; - [Window makeKeyAndOrderFront:nil]; - - displayRect = CGDisplayBounds(display); - ScreenWidth = DeviceWidth = (int)displayRect.size.width; - ScreenHeight = DeviceHeight = (int)displayRect.size.height; - CreationParams.WindowSize.set(ScreenWidth, ScreenHeight); - result = true; - } - } - if (!result) - CGReleaseAllDisplays(); - } - } - } - } - @catch (NSException *exception) - { - closeDevice(); - result = false; - } - - if (result) - { - // fullscreen? - if (Window == NULL && !CreationParams.WindowId) //hide menus in fullscreen mode only -#ifdef __MAC_10_6 - [NSApp setPresentationOptions:(NSApplicationPresentationAutoHideDock | NSApplicationPresentationAutoHideMenuBar)]; -#else - SetSystemUIMode(kUIModeAllHidden, kUIOptionAutoShowMenuBar); -#endif - - if(CreationParams.DriverType == video::EDT_OPENGL) - { - CGLSetCurrentContext(CGLContext); - newSwapInterval = (CreationParams.Vsync) ? 1 : 0; - CGLSetParameter(CGLContext,kCGLCPSwapInterval,&newSwapInterval); - } - } - - return (result); -} - -void CIrrDeviceMacOSX::setResize(int width, int height) -{ - // set new window size - DeviceWidth = width; - DeviceHeight = height; - - // update the size of the opengl rendering context - if(OGLContext); - [OGLContext update]; - - // resize the driver to the inner pane size - if (Window) - { - NSRect driverFrame = [Window contentRectForFrameRect:[Window frame]]; - getVideoDriver()->OnResize(core::dimension2d( (s32)driverFrame.size.width, (s32)driverFrame.size.height)); - } - else - getVideoDriver()->OnResize(core::dimension2d( (s32)width, (s32)height)); - - if (CreationParams.WindowId && OGLContext) - [(NSOpenGLContext *)OGLContext update]; -} - - -void CIrrDeviceMacOSX::createDriver() -{ - switch (CreationParams.DriverType) - { - case video::EDT_SOFTWARE: - #ifdef _IRR_COMPILE_WITH_SOFTWARE_ - VideoDriver = video::createSoftwareDriver(CreationParams.WindowSize, CreationParams.Fullscreen, FileSystem, this); - SoftwareRendererType = 2; - #else - os::Printer::log("No Software driver support compiled in.", ELL_ERROR); - #endif - break; - - case video::EDT_BURNINGSVIDEO: - #ifdef _IRR_COMPILE_WITH_BURNINGSVIDEO_ - VideoDriver = video::createBurningVideoDriver(CreationParams, FileSystem, this); - SoftwareRendererType = 1; - #else - os::Printer::log("Burning's video driver was not compiled in.", ELL_ERROR); - #endif - break; - - case video::EDT_OPENGL: - #ifdef _IRR_COMPILE_WITH_OPENGL_ - VideoDriver = video::createOpenGLDriver(CreationParams, FileSystem, this); - #else - os::Printer::log("No OpenGL support compiled in.", ELL_ERROR); - #endif - break; - - case video::EDT_DIRECT3D8: - case video::EDT_DIRECT3D9: - os::Printer::log("This driver is not available in OSX. Try OpenGL or Software renderer.", ELL_ERROR); - break; - - case video::EDT_NULL: - VideoDriver = video::createNullDriver(FileSystem, CreationParams.WindowSize); - break; - - default: - os::Printer::log("Unable to create video driver of unknown type.", ELL_ERROR); - break; - } -} - -void CIrrDeviceMacOSX::flush() -{ - if (CGLContext != NULL) - { - glFinish(); - CGLFlushDrawable(CGLContext); - } -} - -bool CIrrDeviceMacOSX::run() -{ - NSAutoreleasePool* Pool = [[NSAutoreleasePool alloc] init]; - - NSEvent *event; - irr::SEvent ievent; - - os::Timer::tick(); - storeMouseLocation(); - - event = [NSApp nextEventMatchingMask:NSAnyEventMask untilDate:[NSDate distantPast] inMode:NSDefaultRunLoopMode dequeue:YES]; - if (event != nil) - { - bzero(&ievent,sizeof(ievent)); - - switch([(NSEvent *)event type]) - { - case NSKeyDown: - postKeyEvent(event,ievent,true); - break; - - case NSKeyUp: - postKeyEvent(event,ievent,false); - break; - - case NSFlagsChanged: - ievent.EventType = irr::EET_KEY_INPUT_EVENT; - ievent.KeyInput.Shift = ([(NSEvent *)event modifierFlags] & NSShiftKeyMask) != 0; - ievent.KeyInput.Control = ([(NSEvent *)event modifierFlags] & NSControlKeyMask) != 0; - - if (IsShiftDown != ievent.KeyInput.Shift) - { - ievent.KeyInput.Char = irr::KEY_SHIFT; - ievent.KeyInput.Key = irr::KEY_SHIFT; - ievent.KeyInput.PressedDown = ievent.KeyInput.Shift; - - IsShiftDown = ievent.KeyInput.Shift; - - postEventFromUser(ievent); - } - - if (IsControlDown != ievent.KeyInput.Control) - { - ievent.KeyInput.Char = irr::KEY_CONTROL; - ievent.KeyInput.Key = irr::KEY_CONTROL; - ievent.KeyInput.PressedDown = ievent.KeyInput.Control; - - IsControlDown = ievent.KeyInput.Control; - - postEventFromUser(ievent); - } - - [NSApp sendEvent:event]; - break; - - case NSLeftMouseDown: - ievent.EventType = irr::EET_MOUSE_INPUT_EVENT; - ievent.MouseInput.Event = irr::EMIE_LMOUSE_PRESSED_DOWN; - MouseButtonStates |= irr::EMBSM_LEFT; - ievent.MouseInput.ButtonStates = MouseButtonStates; - postMouseEvent(event,ievent); - break; - - case NSLeftMouseUp: - ievent.EventType = irr::EET_MOUSE_INPUT_EVENT; - MouseButtonStates &= !irr::EMBSM_LEFT; - ievent.MouseInput.ButtonStates = MouseButtonStates; - ievent.MouseInput.Event = irr::EMIE_LMOUSE_LEFT_UP; - postMouseEvent(event,ievent); - break; - - case NSOtherMouseDown: - ievent.EventType = irr::EET_MOUSE_INPUT_EVENT; - ievent.MouseInput.Event = irr::EMIE_MMOUSE_PRESSED_DOWN; - MouseButtonStates |= irr::EMBSM_MIDDLE; - ievent.MouseInput.ButtonStates = MouseButtonStates; - postMouseEvent(event,ievent); - break; - - case NSOtherMouseUp: - ievent.EventType = irr::EET_MOUSE_INPUT_EVENT; - MouseButtonStates &= !irr::EMBSM_MIDDLE; - ievent.MouseInput.ButtonStates = MouseButtonStates; - ievent.MouseInput.Event = irr::EMIE_MMOUSE_LEFT_UP; - postMouseEvent(event,ievent); - break; - - case NSMouseMoved: - case NSLeftMouseDragged: - case NSRightMouseDragged: - case NSOtherMouseDragged: - ievent.EventType = irr::EET_MOUSE_INPUT_EVENT; - ievent.MouseInput.Event = irr::EMIE_MOUSE_MOVED; - ievent.MouseInput.ButtonStates = MouseButtonStates; - postMouseEvent(event,ievent); - break; - - case NSRightMouseDown: - ievent.EventType = irr::EET_MOUSE_INPUT_EVENT; - ievent.MouseInput.Event = irr::EMIE_RMOUSE_PRESSED_DOWN; - MouseButtonStates |= irr::EMBSM_RIGHT; - ievent.MouseInput.ButtonStates = MouseButtonStates; - postMouseEvent(event,ievent); - break; - - case NSRightMouseUp: - ievent.EventType = irr::EET_MOUSE_INPUT_EVENT; - ievent.MouseInput.Event = irr::EMIE_RMOUSE_LEFT_UP; - MouseButtonStates &= !irr::EMBSM_RIGHT; - ievent.MouseInput.ButtonStates = MouseButtonStates; - postMouseEvent(event,ievent); - break; - - case NSScrollWheel: - ievent.EventType = irr::EET_MOUSE_INPUT_EVENT; - ievent.MouseInput.Event = irr::EMIE_MOUSE_WHEEL; - ievent.MouseInput.Wheel = [(NSEvent *)event deltaY]; - if (ievent.MouseInput.Wheel < 1.0f) - ievent.MouseInput.Wheel *= 10.0f; - else - ievent.MouseInput.Wheel *= 5.0f; - postMouseEvent(event,ievent); - break; - - default: - [NSApp sendEvent:event]; - break; - } - } - - pollJoysticks(); - - [Pool release]; - - return (![[NSApp delegate] isQuit] && IsActive); -} - - -//! Pause the current process for the minimum time allowed only to allow other processes to execute -void CIrrDeviceMacOSX::yield() -{ - struct timespec ts = {0,0}; - nanosleep(&ts, NULL); -} - - -//! Pause execution and let other processes to run for a specified amount of time. -void CIrrDeviceMacOSX::sleep(u32 timeMs, bool pauseTimer=false) -{ - bool wasStopped = Timer ? Timer->isStopped() : true; - - struct timespec ts; - ts.tv_sec = (time_t) (timeMs / 1000); - ts.tv_nsec = (long) (timeMs % 1000) * 1000000; - - if (pauseTimer && !wasStopped) - Timer->stop(); - - nanosleep(&ts, NULL); - - if (pauseTimer && !wasStopped) - Timer->start(); -} - - -void CIrrDeviceMacOSX::setWindowCaption(const wchar_t* text) -{ - size_t size; - char title[1024]; - - if (Window != NULL) - { - size = wcstombs(title,text,1024); - title[1023] = 0; -#ifdef __MAC_10_6 - NSString* name = [NSString stringWithCString:title encoding:NSUTF8StringEncoding]; -#else - NSString* name = [NSString stringWithCString:title length:size]; -#endif - [Window setTitle:name]; - [name release]; - } -} - - -bool CIrrDeviceMacOSX::isWindowActive() const -{ - return (IsActive); -} - - -bool CIrrDeviceMacOSX::isWindowFocused() const -{ - if (Window != NULL) - return [Window isKeyWindow]; - return false; -} - - -bool CIrrDeviceMacOSX::isWindowMinimized() const -{ - if (Window != NULL) - return [Window isMiniaturized]; - return false; -} - - -void CIrrDeviceMacOSX::postKeyEvent(void *event,irr::SEvent &ievent,bool pressed) -{ - NSString *str; - std::map::const_iterator iter; - unsigned int result,c,mkey,mchar; - const unsigned char *cStr; - BOOL skipCommand; - - str = [(NSEvent *)event characters]; - if ((str != nil) && ([str length] > 0)) - { - mkey = mchar = 0; - skipCommand = false; - c = [str characterAtIndex:0]; - mchar = c; - - iter = KeyCodes.find([(NSEvent *)event keyCode]); - if (iter != KeyCodes.end()) - mkey = (*iter).second; - else if ((iter = KeyCodes.find(c)) != KeyCodes.end()) - mkey = (*iter).second; - else - { - // workaround for period character - if (c == 0x2E) - { - mkey = irr::KEY_PERIOD; - mchar = '.'; - } - else - { - cStr = (unsigned char *)[str cStringUsingEncoding:NSWindowsCP1252StringEncoding]; - if (cStr != NULL && strlen((char*)cStr) > 0) - { - mchar = cStr[0]; - mkey = toupper(mchar); - if ([(NSEvent *)event modifierFlags] & NSCommandKeyMask) - { - if (mkey == 'C' || mkey == 'V' || mkey == 'X') - { - mchar = 0; - skipCommand = true; - } - } - } - } - } - - ievent.EventType = irr::EET_KEY_INPUT_EVENT; - ievent.KeyInput.Key = (irr::EKEY_CODE)mkey; - ievent.KeyInput.PressedDown = pressed; - ievent.KeyInput.Shift = ([(NSEvent *)event modifierFlags] & NSShiftKeyMask) != 0; - ievent.KeyInput.Control = ([(NSEvent *)event modifierFlags] & NSControlKeyMask) != 0; - ievent.KeyInput.Char = mchar; - - if (skipCommand) - ievent.KeyInput.Control = true; - else if ([(NSEvent *)event modifierFlags] & NSCommandKeyMask) - [NSApp sendEvent:(NSEvent *)event]; - - postEventFromUser(ievent); - } -} - - -void CIrrDeviceMacOSX::postMouseEvent(void *event,irr::SEvent &ievent) -{ - bool post = true; - - if (Window != NULL) - { - ievent.MouseInput.X = (int)[(NSEvent *)event locationInWindow].x; - ievent.MouseInput.Y = DeviceHeight - (int)[(NSEvent *)event locationInWindow].y; - - if (ievent.MouseInput.Y < 0) - post = false; - } - else - { - CGEventRef ourEvent = CGEventCreate(NULL); - CGPoint point = CGEventGetLocation(ourEvent); - CFRelease(ourEvent); - - ievent.MouseInput.X = (int)point.x; - ievent.MouseInput.Y = (int)point.y; - - if (ievent.MouseInput.Y < 0) - post = false; - } - - if (post) - postEventFromUser(ievent); - - [NSApp sendEvent:(NSEvent *)event]; -} - - -void CIrrDeviceMacOSX::storeMouseLocation() -{ - int x,y; - - if (Window != NULL) - { - NSPoint p; - p = [NSEvent mouseLocation]; - p = [Window convertScreenToBase:p]; - x = (int)p.x; - y = DeviceHeight - (int)p.y; - } - else - { - CGEventRef ourEvent = CGEventCreate(NULL); - CGPoint point = CGEventGetLocation(ourEvent); - CFRelease(ourEvent); - - x = (int)point.x; - y = (int)point.y; - - const core::position2di& curr = ((CCursorControl *)CursorControl)->getPosition(); - if (curr.X != x || curr.Y != y) - { - // In fullscreen mode, events are not sent regularly so rely on polling - irr::SEvent ievent; - ievent.EventType = irr::EET_MOUSE_INPUT_EVENT; - ievent.MouseInput.Event = irr::EMIE_MOUSE_MOVED; - ievent.MouseInput.X = x; - ievent.MouseInput.Y = y; - postEventFromUser(ievent); - } - } - - ((CCursorControl *)CursorControl)->updateInternalCursorPosition(x,y); -} - - -void CIrrDeviceMacOSX::setMouseLocation(int x,int y) -{ - NSPoint p; - CGPoint c; - - if (Window != NULL) - { - // Irrlicht window exists - p.x = (float) x; - p.y = (float) (DeviceHeight - y); - p = [Window convertBaseToScreen:p]; - p.y = ScreenHeight - p.y; - } - else - { - p.x = (float) x; - p.y = (float) y + (ScreenHeight - DeviceHeight); - } - - c.x = p.x; - c.y = p.y; - -#ifdef __MAC_10_6 - /*CGEventSourceRef SourceRef = CGEventSourceCreate(0); - CGEventSourceSetLocalEventsSuppressionInterval(SourceRef, 0); - CFRelease(SourceRef);*/ - CGSetLocalEventsSuppressionInterval(0); -#else - CGSetLocalEventsSuppressionInterval(0); -#endif - CGWarpMouseCursorPosition(c); -} - - -void CIrrDeviceMacOSX::setCursorVisible(bool visible) -{ - if (visible) - CGDisplayShowCursor(CGMainDisplayID()); - else - CGDisplayHideCursor(CGMainDisplayID()); -} - - -void CIrrDeviceMacOSX::initKeycodes() -{ - KeyCodes[kVK_UpArrow] = irr::KEY_UP; - KeyCodes[kVK_DownArrow] = irr::KEY_DOWN; - KeyCodes[kVK_LeftArrow] = irr::KEY_LEFT; - KeyCodes[kVK_RightArrow] = irr::KEY_RIGHT; - KeyCodes[kVK_F1] = irr::KEY_F1; - KeyCodes[kVK_F2] = irr::KEY_F2; - KeyCodes[kVK_F3] = irr::KEY_F3; - KeyCodes[kVK_F4] = irr::KEY_F4; - KeyCodes[kVK_F5] = irr::KEY_F5; - KeyCodes[kVK_F6] = irr::KEY_F6; - KeyCodes[kVK_F7] = irr::KEY_F7; - KeyCodes[kVK_F8] = irr::KEY_F8; - KeyCodes[kVK_F9] = irr::KEY_F9; - KeyCodes[kVK_F10] = irr::KEY_F10; - KeyCodes[kVK_F11] = irr::KEY_F11; - KeyCodes[kVK_F12] = irr::KEY_F12; - KeyCodes[kVK_F13] = irr::KEY_F13; - KeyCodes[kVK_F14] = irr::KEY_F14; - KeyCodes[kVK_F15] = irr::KEY_F15; - KeyCodes[kVK_F16] = irr::KEY_F16; - KeyCodes[kVK_F17] = irr::KEY_F17; - KeyCodes[kVK_F18] = irr::KEY_F18; - KeyCodes[kVK_F19] = irr::KEY_F19; - KeyCodes[kVK_F20] = irr::KEY_F20; - KeyCodes[kVK_Home] = irr::KEY_HOME; - KeyCodes[kVK_End] = irr::KEY_END; - KeyCodes[NSInsertFunctionKey] = irr::KEY_INSERT; - KeyCodes[kVK_ForwardDelete] = irr::KEY_DELETE; - KeyCodes[kVK_Help] = irr::KEY_HELP; - KeyCodes[NSSelectFunctionKey] = irr::KEY_SELECT; - KeyCodes[NSPrintFunctionKey] = irr::KEY_PRINT; - KeyCodes[NSExecuteFunctionKey] = irr::KEY_EXECUT; - KeyCodes[NSPrintScreenFunctionKey] = irr::KEY_SNAPSHOT; - KeyCodes[NSPauseFunctionKey] = irr::KEY_PAUSE; - KeyCodes[NSScrollLockFunctionKey] = irr::KEY_SCROLL; - KeyCodes[kVK_Delete] = irr::KEY_BACK; - KeyCodes[kVK_Tab] = irr::KEY_TAB; - KeyCodes[kVK_Return] = irr::KEY_RETURN; - KeyCodes[kVK_Escape] = irr::KEY_ESCAPE; - KeyCodes[kVK_Control] = irr::KEY_CONTROL; - KeyCodes[kVK_RightControl] = irr::KEY_RCONTROL; - KeyCodes[kVK_Command] = irr::KEY_MENU; - KeyCodes[kVK_Shift] = irr::KEY_SHIFT; - KeyCodes[kVK_RightShift] = irr::KEY_RSHIFT; - KeyCodes[kVK_Space] = irr::KEY_SPACE; - - KeyCodes[kVK_ANSI_A] = irr::KEY_KEY_A; - KeyCodes[kVK_ANSI_B] = irr::KEY_KEY_B; - KeyCodes[kVK_ANSI_C] = irr::KEY_KEY_C; - KeyCodes[kVK_ANSI_D] = irr::KEY_KEY_D; - KeyCodes[kVK_ANSI_E] = irr::KEY_KEY_E; - KeyCodes[kVK_ANSI_F] = irr::KEY_KEY_F; - KeyCodes[kVK_ANSI_G] = irr::KEY_KEY_G; - KeyCodes[kVK_ANSI_H] = irr::KEY_KEY_H; - KeyCodes[kVK_ANSI_I] = irr::KEY_KEY_I; - KeyCodes[kVK_ANSI_J] = irr::KEY_KEY_J; - KeyCodes[kVK_ANSI_K] = irr::KEY_KEY_K; - KeyCodes[kVK_ANSI_L] = irr::KEY_KEY_L; - KeyCodes[kVK_ANSI_M] = irr::KEY_KEY_M; - KeyCodes[kVK_ANSI_N] = irr::KEY_KEY_N; - KeyCodes[kVK_ANSI_O] = irr::KEY_KEY_O; - KeyCodes[kVK_ANSI_P] = irr::KEY_KEY_P; - KeyCodes[kVK_ANSI_Q] = irr::KEY_KEY_Q; - KeyCodes[kVK_ANSI_R] = irr::KEY_KEY_R; - KeyCodes[kVK_ANSI_S] = irr::KEY_KEY_S; - KeyCodes[kVK_ANSI_T] = irr::KEY_KEY_T; - KeyCodes[kVK_ANSI_U] = irr::KEY_KEY_U; - KeyCodes[kVK_ANSI_V] = irr::KEY_KEY_V; - KeyCodes[kVK_ANSI_W] = irr::KEY_KEY_W; - KeyCodes[kVK_ANSI_X] = irr::KEY_KEY_X; - KeyCodes[kVK_ANSI_X] = irr::KEY_KEY_X; - KeyCodes[kVK_ANSI_Y] = irr::KEY_KEY_Y; - KeyCodes[kVK_ANSI_Z] = irr::KEY_KEY_Z; - - KeyCodes[kVK_ANSI_0] = irr::KEY_KEY_0; - KeyCodes[kVK_ANSI_1] = irr::KEY_KEY_1; - KeyCodes[kVK_ANSI_2] = irr::KEY_KEY_2; - KeyCodes[kVK_ANSI_3] = irr::KEY_KEY_3; - KeyCodes[kVK_ANSI_4] = irr::KEY_KEY_4; - KeyCodes[kVK_ANSI_5] = irr::KEY_KEY_5; - KeyCodes[kVK_ANSI_6] = irr::KEY_KEY_6; - KeyCodes[kVK_ANSI_7] = irr::KEY_KEY_7; - KeyCodes[kVK_ANSI_8] = irr::KEY_KEY_8; - KeyCodes[kVK_ANSI_9] = irr::KEY_KEY_9; - - KeyCodes[kVK_ANSI_Slash] = irr::KEY_DIVIDE; - KeyCodes[kVK_ANSI_Comma] = irr::KEY_COMMA; - KeyCodes[kVK_ANSI_Period] = irr::KEY_PERIOD; - KeyCodes[kVK_PageUp] = irr::KEY_PRIOR; - KeyCodes[kVK_PageDown] = irr::KEY_NEXT; - - KeyCodes[kVK_ANSI_Keypad0] = irr::KEY_NUMPAD0; - KeyCodes[kVK_ANSI_Keypad1] = irr::KEY_NUMPAD1; - KeyCodes[kVK_ANSI_Keypad2] = irr::KEY_NUMPAD2; - KeyCodes[kVK_ANSI_Keypad3] = irr::KEY_NUMPAD3; - KeyCodes[kVK_ANSI_Keypad4] = irr::KEY_NUMPAD4; - KeyCodes[kVK_ANSI_Keypad5] = irr::KEY_NUMPAD5; - KeyCodes[kVK_ANSI_Keypad6] = irr::KEY_NUMPAD6; - KeyCodes[kVK_ANSI_Keypad7] = irr::KEY_NUMPAD7; - KeyCodes[kVK_ANSI_Keypad8] = irr::KEY_NUMPAD8; - KeyCodes[kVK_ANSI_Keypad9] = irr::KEY_NUMPAD9; - - KeyCodes[kVK_ANSI_KeypadDecimal] = irr::KEY_DECIMAL; - KeyCodes[kVK_ANSI_KeypadMultiply] = irr::KEY_MULTIPLY; - KeyCodes[kVK_ANSI_KeypadPlus] = irr::KEY_PLUS; - KeyCodes[kVK_ANSI_KeypadClear] = irr::KEY_OEM_CLEAR; - KeyCodes[kVK_ANSI_KeypadDivide] = irr::KEY_DIVIDE; - KeyCodes[kVK_ANSI_KeypadEnter] = irr::KEY_RETURN; - KeyCodes[kVK_ANSI_KeypadMinus] = irr::KEY_SUBTRACT; -} - - -//! Sets if the window should be resizable in windowed mode. -void CIrrDeviceMacOSX::setResizable(bool resize) -{ - IsResizable = resize; -#if 0 - if (resize) - [Window setStyleMask:NSTitledWindowMask|NSClosableWindowMask|NSMiniaturizableWindowMask|NSResizableWindowMask]; - else - [Window setStyleMask:NSTitledWindowMask|NSClosableWindowMask]; -#endif -} - - -bool CIrrDeviceMacOSX::isResizable() const -{ - return IsResizable; -} - - -void CIrrDeviceMacOSX::minimizeWindow() -{ - if (Window != NULL) - [Window miniaturize:[NSApp self]]; -} - - -//! Maximizes the window if possible. -void CIrrDeviceMacOSX::maximizeWindow() -{ - // todo: implement -} - - -//! Restore the window to normal size if possible. -void CIrrDeviceMacOSX::restoreWindow() -{ - [Window deminiaturize:[NSApp self]]; -} - - -bool CIrrDeviceMacOSX::present(video::IImage* surface, void* windowId, core::rect* src ) -{ - // todo: implement window ID and src rectangle - - if (!surface) - return false; - - if (SoftwareRendererType > 0) - { - const u32 colorSamples=3; - // do we need to change the size? - const bool updateSize = !SoftwareDriverTarget || - s32([SoftwareDriverTarget size].width) != surface->getDimension().Width || - s32([SoftwareDriverTarget size].height) != surface->getDimension().Height; - - NSRect areaRect = NSMakeRect(0.0, 0.0, surface->getDimension().Width, surface->getDimension().Height); - const u32 destPitch = (colorSamples * areaRect.size.width); - - // create / update the target - if (updateSize) - { - [SoftwareDriverTarget release]; - // allocate target for IImage - SoftwareDriverTarget = [[NSBitmapImageRep alloc] - initWithBitmapDataPlanes: nil - pixelsWide: areaRect.size.width - pixelsHigh: areaRect.size.height - bitsPerSample: 8 - samplesPerPixel: colorSamples - hasAlpha: NO - isPlanar: NO - colorSpaceName: NSCalibratedRGBColorSpace - bytesPerRow: destPitch - bitsPerPixel: 8*colorSamples]; - } - - if (SoftwareDriverTarget==nil) - return false; - - // get pointer to image data - unsigned char* imgData = (unsigned char*)surface->lock(); - - u8* srcdata = reinterpret_cast(imgData); - u8* destData = reinterpret_cast([SoftwareDriverTarget bitmapData]); - const u32 srcheight = core::min_(surface->getDimension().Height, (u32)areaRect.size.height); - const u32 srcPitch = surface->getPitch(); - const u32 minWidth = core::min_(surface->getDimension().Width, (u32)areaRect.size.width); - for (u32 y=0; y!=srcheight; ++y) - { - if(SoftwareRendererType == 2) - { - if (surface->getColorFormat() == video::ECF_A8R8G8B8) - video::CColorConverter::convert_A8R8G8B8toB8G8R8(srcdata, minWidth, destData); - else if (surface->getColorFormat() == video::ECF_A1R5G5B5) - video::CColorConverter::convert_A1R5G5B5toB8G8R8(srcdata, minWidth, destData); - else - video::CColorConverter::convert_viaFormat(srcdata, surface->getColorFormat(), minWidth, destData, video::ECF_R8G8B8); - } - else - { - if (surface->getColorFormat() == video::ECF_A8R8G8B8) - video::CColorConverter::convert_A8R8G8B8toR8G8B8(srcdata, minWidth, destData); - else if (surface->getColorFormat() == video::ECF_A1R5G5B5) - video::CColorConverter::convert_A1R5G5B5toR8G8B8(srcdata, minWidth, destData); - else - video::CColorConverter::convert_viaFormat(srcdata, surface->getColorFormat(), minWidth, destData, video::ECF_R8G8B8); - } - - srcdata += srcPitch; - destData += destPitch; - } - - // unlock the data - surface->unlock(); - - // todo: draw properly into a sub-view - [SoftwareDriverTarget draw]; - } - - return false; -} - - -#if defined (_IRR_COMPILE_WITH_JOYSTICK_EVENTS_) -static void joystickRemovalCallback(void * target, - IOReturn result, void * refcon, void * sender) -{ - JoystickInfo *joy = (JoystickInfo *) refcon; - joy->removed = 1; -} -#endif // _IRR_COMPILE_WITH_JOYSTICK_EVENTS_ - - -bool CIrrDeviceMacOSX::activateJoysticks(core::array & joystickInfo) -{ -#if defined (_IRR_COMPILE_WITH_JOYSTICK_EVENTS_) - ActiveJoysticks.clear(); - joystickInfo.clear(); - - io_object_t hidObject = 0; - io_iterator_t hidIterator = 0; - IOReturn result = kIOReturnSuccess; - mach_port_t masterPort = 0; - CFMutableDictionaryRef hidDictionaryRef = NULL; - - result = IOMasterPort (bootstrap_port, &masterPort); - if (kIOReturnSuccess != result) - { - os::Printer::log("initialiseJoysticks IOMasterPort failed", ELL_ERROR); - return false; - } - - hidDictionaryRef = IOServiceMatching (kIOHIDDeviceKey); - if (!hidDictionaryRef) - { - os::Printer::log("initialiseJoysticks IOServiceMatching failed", ELL_ERROR); - return false; - } - result = IOServiceGetMatchingServices (masterPort, hidDictionaryRef, &hidIterator); - - if (kIOReturnSuccess != result) - { - os::Printer::log("initialiseJoysticks IOServiceGetMatchingServices failed", ELL_ERROR); - return false; - } - - //no joysticks just return - if (!hidIterator) - return false; - - u32 jindex = 0u; - while ((hidObject = IOIteratorNext (hidIterator))) - { - JoystickInfo info; - - // get dictionary for HID properties - CFMutableDictionaryRef hidProperties = 0; - - kern_return_t kern_result = IORegistryEntryCreateCFProperties (hidObject, &hidProperties, kCFAllocatorDefault, kNilOptions); - if ((kern_result == KERN_SUCCESS) && hidProperties) - { - HRESULT plugInResult = S_OK; - SInt32 score = 0; - IOCFPlugInInterface ** ppPlugInInterface = NULL; - result = IOCreatePlugInInterfaceForService (hidObject, kIOHIDDeviceUserClientTypeID, - kIOCFPlugInInterfaceID, &ppPlugInInterface, &score); - if (kIOReturnSuccess == result) - { - plugInResult = (*ppPlugInInterface)->QueryInterface (ppPlugInInterface, - CFUUIDGetUUIDBytes (kIOHIDDeviceInterfaceID), (void **) &(info.interface)); - if (plugInResult != S_OK) - os::Printer::log("initialiseJoysticks query HID class device interface failed", ELL_ERROR); - (*ppPlugInInterface)->Release(ppPlugInInterface); - } - else - continue; - - if (info.interface != NULL) - { - result = (*(info.interface))->open (info.interface, 0); - if (result == kIOReturnSuccess) - { - (*(info.interface))->setRemovalCallback (info.interface, joystickRemovalCallback, &info, &info); - getJoystickDeviceInfo(hidObject, hidProperties, &info); - - // get elements - CFTypeRef refElementTop = CFDictionaryGetValue (hidProperties, CFSTR(kIOHIDElementKey)); - if (refElementTop) - { - CFTypeID type = CFGetTypeID (refElementTop); - if (type == CFArrayGetTypeID()) - { - CFRange range = {0, CFArrayGetCount ((CFArrayRef)refElementTop)}; - info.numActiveJoysticks = ActiveJoysticks.size(); - CFArrayApplyFunction ((CFArrayRef)refElementTop, range, getJoystickComponentArrayHandler, &info); - } - } - } - else - { - CFRelease (hidProperties); - os::Printer::log("initialiseJoysticks Open interface failed", ELL_ERROR); - continue; - } - - CFRelease (hidProperties); - - result = IOObjectRelease (hidObject); - - if ( (info.usagePage != kHIDPage_GenericDesktop) || - ((info.usage != kHIDUsage_GD_Joystick && - info.usage != kHIDUsage_GD_GamePad && - info.usage != kHIDUsage_GD_MultiAxisController)) ) - { - closeJoystickDevice (&info); - continue; - } - - for (u32 i = 0; i < 6; ++i) - info.persistentData.JoystickEvent.Axis[i] = 0; - - ActiveJoysticks.push_back(info); - - SJoystickInfo returnInfo; - returnInfo.Joystick = jindex; - returnInfo.Axes = info.axes; - //returnInfo.Hats = info.hats; - returnInfo.Buttons = info.buttons; - returnInfo.Name = info.joystickName; - returnInfo.PovHat = SJoystickInfo::POV_HAT_UNKNOWN; - ++ jindex; - - //if (info.hatComp.size()) - // returnInfo.PovHat = SJoystickInfo::POV_HAT_PRESENT; - //else - // returnInfo.PovHat = SJoystickInfo::POV_HAT_ABSENT; - - joystickInfo.push_back(returnInfo); - } - - } - else - { - continue; - } - } - result = IOObjectRelease (hidIterator); - - return true; -#endif // _IRR_COMPILE_WITH_JOYSTICK_EVENTS_ - - return false; -} - -void CIrrDeviceMacOSX::pollJoysticks() -{ -#if defined (_IRR_COMPILE_WITH_JOYSTICK_EVENTS_) - if(0 == ActiveJoysticks.size()) - return; - - u32 joystick; - for (joystick = 0; joystick < ActiveJoysticks.size(); ++joystick) - { - if (ActiveJoysticks[joystick].removed) - continue; - - bool found = false; - ActiveJoysticks[joystick].persistentData.JoystickEvent.Joystick = joystick; - - if (ActiveJoysticks[joystick].interface) - { - for (u32 n = 0; n < ActiveJoysticks[joystick].axisComp.size(); n++) - { - IOReturn result = kIOReturnSuccess; - IOHIDEventStruct hidEvent; - hidEvent.value = 0; - result = (*(ActiveJoysticks[joystick].interface))->getElementValue(ActiveJoysticks[joystick].interface, ActiveJoysticks[joystick].axisComp[n].cookie, &hidEvent); - if (kIOReturnSuccess == result) - { - const f32 min = -32768.0f; - const f32 max = 32767.0f; - const f32 deviceScale = max - min; - const f32 readScale = (f32)ActiveJoysticks[joystick].axisComp[n].maxRead - (f32)ActiveJoysticks[joystick].axisComp[n].minRead; - - if (hidEvent.value < ActiveJoysticks[joystick].axisComp[n].minRead) - ActiveJoysticks[joystick].axisComp[n].minRead = hidEvent.value; - if (hidEvent.value > ActiveJoysticks[joystick].axisComp[n].maxRead) - ActiveJoysticks[joystick].axisComp[n].maxRead = hidEvent.value; - - if (readScale != 0.0f) - hidEvent.value = (int)(((f32)((f32)hidEvent.value - (f32)ActiveJoysticks[joystick].axisComp[n].minRead) * deviceScale / readScale) + min); - - if (ActiveJoysticks[joystick].persistentData.JoystickEvent.Axis[n] != (s16)hidEvent.value) - found = true; - ActiveJoysticks[joystick].persistentData.JoystickEvent.Axis[n] = (s16)hidEvent.value; - } - }//axis check - - for (u32 n = 0; n < ActiveJoysticks[joystick].buttonComp.size(); n++) - { - IOReturn result = kIOReturnSuccess; - IOHIDEventStruct hidEvent; - hidEvent.value = 0; - result = (*(ActiveJoysticks[joystick].interface))->getElementValue(ActiveJoysticks[joystick].interface, ActiveJoysticks[joystick].buttonComp[n].cookie, &hidEvent); - if (kIOReturnSuccess == result) - { - u32 ButtonStates = 0; - - if (hidEvent.value && !((ActiveJoysticks[joystick].persistentData.JoystickEvent.ButtonStates & (1 << n)) ? true : false) ) - found = true; - else if (!hidEvent.value && ((ActiveJoysticks[joystick].persistentData.JoystickEvent.ButtonStates & (1 << n)) ? true : false)) - found = true; - - if (hidEvent.value) - ActiveJoysticks[joystick].persistentData.JoystickEvent.ButtonStates |= (1 << n); - else - ActiveJoysticks[joystick].persistentData.JoystickEvent.ButtonStates &= ~(1 << n); - } - }//button check - //still ToDo..will be done soon :) -/* - for (u32 n = 0; n < ActiveJoysticks[joystick].hatComp.size(); n++) - { - IOReturn result = kIOReturnSuccess; - IOHIDEventStruct hidEvent; - hidEvent.value = 0; - result = (*(ActiveJoysticks[joystick].interface))->getElementValue(ActiveJoysticks[joystick].interface, ActiveJoysticks[joystick].hatComp[n].cookie, &hidEvent); - if (kIOReturnSuccess == result) - { - if (ActiveJoysticks[joystick].persistentData.JoystickEvent.POV != hidEvent.value) - found = true; - ActiveJoysticks[joystick].persistentData.JoystickEvent.POV = hidEvent.value; - } - }//hat check -*/ - } - - if (found) - postEventFromUser(ActiveJoysticks[joystick].persistentData); - } -#endif // _IRR_COMPILE_WITH_JOYSTICK_EVENTS_ -} - -video::IVideoModeList* CIrrDeviceMacOSX::getVideoModeList() -{ - if (!VideoModeList.getVideoModeCount()) - { - CGDirectDisplayID display; - display = CGMainDisplayID(); - -#ifdef __MAC_10_6 - CFArrayRef Modes = CGDisplayCopyAllDisplayModes(display, NULL); - - for(int i = 0; i < CFArrayGetCount(Modes); ++i) - { - CGDisplayModeRef CurrentMode = (CGDisplayModeRef)CFArrayGetValueAtIndex(Modes, i); - - u8 Depth = 0; - - CFStringRef pixEnc = CGDisplayModeCopyPixelEncoding(CurrentMode); - - if(CFStringCompare(pixEnc, CFSTR(IO32BitDirectPixels), kCFCompareCaseInsensitive) == kCFCompareEqualTo) - Depth = 32; - else - if(CFStringCompare(pixEnc, CFSTR(IO16BitDirectPixels), kCFCompareCaseInsensitive) == kCFCompareEqualTo) - Depth = 16; - else - if(CFStringCompare(pixEnc, CFSTR(IO8BitIndexedPixels), kCFCompareCaseInsensitive) == kCFCompareEqualTo) - Depth = 8; - - if(Depth) - { - unsigned int Width = CGDisplayModeGetWidth(CurrentMode); - unsigned int Height = CGDisplayModeGetHeight(CurrentMode); - - VideoModeList.addMode(core::dimension2d(Width, Height), Depth); - } - } -#else - CFArrayRef availableModes = CGDisplayAvailableModes(display); - unsigned int numberOfAvailableModes = CFArrayGetCount(availableModes); - for (u32 i= 0; i(width, height), - bitsPerPixel); - } -#endif - } - return &VideoModeList; -} - -} // end namespace - -#endif // _IRR_COMPILE_WITH_OSX_DEVICE_ - +// Copyright (C) 2005-2006 Etienne Petitjean +// Copyright (C) 2007-2012 Christian Stehno +// This file is part of the "Irrlicht Engine". +// For conditions of distribution and use, see copyright notice in Irrlicht.h + +#include "IrrCompileConfig.h" + +#ifdef _IRR_COMPILE_WITH_OSX_DEVICE_ + +#import +#import +#ifndef __MAC_10_6 +#import +#endif + +#include "CIrrDeviceMacOSX.h" +#include "IEventReceiver.h" +#include "irrList.h" +#include "os.h" +#include "CTimer.h" +#include "irrString.h" +#include "Keycodes.h" +#include +#include +#include "COSOperator.h" +#include "CColorConverter.h" +#include "irrlicht.h" + + +#import +#import +#import "AppDelegate.h" + +#if defined _IRR_COMPILE_WITH_JOYSTICK_EVENTS_ + +#include +#include +#ifdef MACOS_10_0_4 +#include +#else +/* The header was moved here in Mac OS X 10.1 */ +#include +#endif +#include +#include + +// only OSX 10.5 seems to not need these defines... +#if !defined(__MAC_10_5) || defined(__MAC_10_6) +// Contents from Events.h from Carbon/HIToolbox but we need it with Cocoa too +// and for some reason no Cocoa equivalent of these constants seems provided. +// So I'm doing like everyone else and using copy-and-paste. + +/* + * Summary: + * Virtual keycodes + * + * Discussion: + * These constants are the virtual keycodes defined originally in + * Inside Mac Volume V, pg. V-191. They identify physical keys on a + * keyboard. Those constants with "ANSI" in the name are labeled + * according to the key position on an ANSI-standard US keyboard. + * For example, kVK_ANSI_A indicates the virtual keycode for the key + * with the letter 'A' in the US keyboard layout. Other keyboard + * layouts may have the 'A' key label on a different physical key; + * in this case, pressing 'A' will generate a different virtual + * keycode. + */ +enum { + kVK_ANSI_A = 0x00, + kVK_ANSI_S = 0x01, + kVK_ANSI_D = 0x02, + kVK_ANSI_F = 0x03, + kVK_ANSI_H = 0x04, + kVK_ANSI_G = 0x05, + kVK_ANSI_Z = 0x06, + kVK_ANSI_X = 0x07, + kVK_ANSI_C = 0x08, + kVK_ANSI_V = 0x09, + kVK_ANSI_B = 0x0B, + kVK_ANSI_Q = 0x0C, + kVK_ANSI_W = 0x0D, + kVK_ANSI_E = 0x0E, + kVK_ANSI_R = 0x0F, + kVK_ANSI_Y = 0x10, + kVK_ANSI_T = 0x11, + kVK_ANSI_1 = 0x12, + kVK_ANSI_2 = 0x13, + kVK_ANSI_3 = 0x14, + kVK_ANSI_4 = 0x15, + kVK_ANSI_6 = 0x16, + kVK_ANSI_5 = 0x17, + kVK_ANSI_Equal = 0x18, + kVK_ANSI_9 = 0x19, + kVK_ANSI_7 = 0x1A, + kVK_ANSI_Minus = 0x1B, + kVK_ANSI_8 = 0x1C, + kVK_ANSI_0 = 0x1D, + kVK_ANSI_RightBracket = 0x1E, + kVK_ANSI_O = 0x1F, + kVK_ANSI_U = 0x20, + kVK_ANSI_LeftBracket = 0x21, + kVK_ANSI_I = 0x22, + kVK_ANSI_P = 0x23, + kVK_ANSI_L = 0x25, + kVK_ANSI_J = 0x26, + kVK_ANSI_Quote = 0x27, + kVK_ANSI_K = 0x28, + kVK_ANSI_Semicolon = 0x29, + kVK_ANSI_Backslash = 0x2A, + kVK_ANSI_Comma = 0x2B, + kVK_ANSI_Slash = 0x2C, + kVK_ANSI_N = 0x2D, + kVK_ANSI_M = 0x2E, + kVK_ANSI_Period = 0x2F, + kVK_ANSI_Grave = 0x32, + kVK_ANSI_KeypadDecimal = 0x41, + kVK_ANSI_KeypadMultiply = 0x43, + kVK_ANSI_KeypadPlus = 0x45, + kVK_ANSI_KeypadClear = 0x47, + kVK_ANSI_KeypadDivide = 0x4B, + kVK_ANSI_KeypadEnter = 0x4C, + kVK_ANSI_KeypadMinus = 0x4E, + kVK_ANSI_KeypadEquals = 0x51, + kVK_ANSI_Keypad0 = 0x52, + kVK_ANSI_Keypad1 = 0x53, + kVK_ANSI_Keypad2 = 0x54, + kVK_ANSI_Keypad3 = 0x55, + kVK_ANSI_Keypad4 = 0x56, + kVK_ANSI_Keypad5 = 0x57, + kVK_ANSI_Keypad6 = 0x58, + kVK_ANSI_Keypad7 = 0x59, + kVK_ANSI_Keypad8 = 0x5B, + kVK_ANSI_Keypad9 = 0x5C +}; + +/* keycodes for keys that are independent of keyboard layout*/ +enum { + kVK_Return = 0x24, + kVK_Tab = 0x30, + kVK_Space = 0x31, + kVK_Delete = 0x33, + kVK_Escape = 0x35, + kVK_Command = 0x37, + kVK_Shift = 0x38, + kVK_CapsLock = 0x39, + kVK_Option = 0x3A, + kVK_Control = 0x3B, + kVK_RightShift = 0x3C, + kVK_RightOption = 0x3D, + kVK_RightControl = 0x3E, + kVK_Function = 0x3F, + kVK_F17 = 0x40, + kVK_VolumeUp = 0x48, + kVK_VolumeDown = 0x49, + kVK_Mute = 0x4A, + kVK_F18 = 0x4F, + kVK_F19 = 0x50, + kVK_F20 = 0x5A, + kVK_F5 = 0x60, + kVK_F6 = 0x61, + kVK_F7 = 0x62, + kVK_F3 = 0x63, + kVK_F8 = 0x64, + kVK_F9 = 0x65, + kVK_F11 = 0x67, + kVK_F13 = 0x69, + kVK_F16 = 0x6A, + kVK_F14 = 0x6B, + kVK_F10 = 0x6D, + kVK_F12 = 0x6F, + kVK_F15 = 0x71, + kVK_Help = 0x72, + kVK_Home = 0x73, + kVK_PageUp = 0x74, + kVK_ForwardDelete = 0x75, + kVK_F4 = 0x76, + kVK_End = 0x77, + kVK_F2 = 0x78, + kVK_PageDown = 0x79, + kVK_F1 = 0x7A, + kVK_LeftArrow = 0x7B, + kVK_RightArrow = 0x7C, + kVK_DownArrow = 0x7D, + kVK_UpArrow = 0x7E +}; +#endif + +struct JoystickComponent +{ + IOHIDElementCookie cookie; // unique value which identifies element, will NOT change + long min; // reported min value possible + long max; // reported max value possible + + long minRead; //min read value + long maxRead; //max read value + + JoystickComponent() : min(0), minRead(0), max(0), maxRead(0) + { + } +}; + +struct JoystickInfo +{ + irr::core::array axisComp; + irr::core::array buttonComp; + irr::core::array hatComp; + + int hats; + int axes; + int buttons; + int numActiveJoysticks; + + irr::SEvent persistentData; + + IOHIDDeviceInterface ** interface; + bool removed; + char joystickName[256]; + long usage; // usage page from IOUSBHID Parser.h which defines general usage + long usagePage; // usage within above page from IOUSBHID Parser.h which defines specific usage + + JoystickInfo() : hats(0), axes(0), buttons(0), interface(0), removed(false), usage(0), usagePage(0), numActiveJoysticks(0) + { + interface = NULL; + memset(joystickName, '\0', 256); + axisComp.clear(); + buttonComp.clear(); + hatComp.clear(); + + persistentData.EventType = irr::EET_JOYSTICK_INPUT_EVENT; + persistentData.JoystickEvent.POV = 65535; + persistentData.JoystickEvent.ButtonStates = 0; + } +}; +irr::core::array ActiveJoysticks; + +//helper functions for init joystick +static IOReturn closeJoystickDevice (JoystickInfo* joyInfo) +{ + IOReturn result = kIOReturnSuccess; + if (joyInfo && joyInfo->interface) + { + /* close the interface */ + result = (*(joyInfo->interface))->close (joyInfo->interface); + if (kIOReturnNotOpen == result) + { + /* do nothing as device was not opened, thus can't be closed */ + } + else if (kIOReturnSuccess != result) + irr::os::Printer::log("IOHIDDeviceInterface failed to close", irr::ELL_ERROR); + /* release the interface */ + result = (*(joyInfo->interface))->Release (joyInfo->interface); + if (kIOReturnSuccess != result) + irr::os::Printer::log("IOHIDDeviceInterface failed to release", irr::ELL_ERROR); + joyInfo->interface = NULL; + } + return result; +} + +static void addComponentInfo (CFTypeRef refElement, JoystickComponent *pComponent, int numActiveJoysticks) +{ + long number; + CFTypeRef refType; + + refType = CFDictionaryGetValue ((CFDictionaryRef)refElement, CFSTR(kIOHIDElementCookieKey)); + if (refType && CFNumberGetValue ((CFNumberRef)refType, kCFNumberLongType, &number)) + pComponent->cookie = (IOHIDElementCookie) number; + refType = CFDictionaryGetValue ((CFDictionaryRef)refElement, CFSTR(kIOHIDElementMinKey)); + if (refType && CFNumberGetValue ((CFNumberRef)refType, kCFNumberLongType, &number)) + pComponent->minRead = pComponent->min = number; + refType = CFDictionaryGetValue ((CFDictionaryRef)refElement, CFSTR(kIOHIDElementMaxKey)); + if (refType && CFNumberGetValue ((CFNumberRef)refType, kCFNumberLongType, &number)) + pComponent->maxRead = pComponent->max = number; +} + +static void getJoystickComponentArrayHandler (const void * value, void * parameter); + +static void addJoystickComponent (CFTypeRef refElement, JoystickInfo* joyInfo) +{ + long elementType, usagePage, usage; + CFTypeRef refElementType = CFDictionaryGetValue ((CFDictionaryRef)refElement, CFSTR(kIOHIDElementTypeKey)); + CFTypeRef refUsagePage = CFDictionaryGetValue ((CFDictionaryRef)refElement, CFSTR(kIOHIDElementUsagePageKey)); + CFTypeRef refUsage = CFDictionaryGetValue ((CFDictionaryRef)refElement, CFSTR(kIOHIDElementUsageKey)); + + if ((refElementType) && (CFNumberGetValue ((CFNumberRef)refElementType, kCFNumberLongType, &elementType))) + { + /* look at types of interest */ + if ((elementType == kIOHIDElementTypeInput_Misc) || (elementType == kIOHIDElementTypeInput_Button) || + (elementType == kIOHIDElementTypeInput_Axis)) + { + if (refUsagePage && CFNumberGetValue ((CFNumberRef)refUsagePage, kCFNumberLongType, &usagePage) && + refUsage && CFNumberGetValue ((CFNumberRef)refUsage, kCFNumberLongType, &usage)) + { + switch (usagePage) /* only interested in kHIDPage_GenericDesktop and kHIDPage_Button */ + { + case kHIDPage_GenericDesktop: + { + switch (usage) /* look at usage to determine function */ + { + case kHIDUsage_GD_X: + case kHIDUsage_GD_Y: + case kHIDUsage_GD_Z: + case kHIDUsage_GD_Rx: + case kHIDUsage_GD_Ry: + case kHIDUsage_GD_Rz: + case kHIDUsage_GD_Slider: + case kHIDUsage_GD_Dial: + case kHIDUsage_GD_Wheel: + { + joyInfo->axes++; + JoystickComponent newComponent; + addComponentInfo(refElement, &newComponent, joyInfo->numActiveJoysticks); + joyInfo->axisComp.push_back(newComponent); + } + break; + case kHIDUsage_GD_Hatswitch: + { + joyInfo->hats++; + JoystickComponent newComponent; + addComponentInfo(refElement, &newComponent, joyInfo->numActiveJoysticks); + joyInfo->hatComp.push_back(newComponent); + } + break; + } + } + break; + case kHIDPage_Button: + { + joyInfo->buttons++; + JoystickComponent newComponent; + addComponentInfo(refElement, &newComponent, joyInfo->numActiveJoysticks); + joyInfo->buttonComp.push_back(newComponent); + } + break; + default: + break; + } + } + } + else if (kIOHIDElementTypeCollection == elementType) + { + //get elements + CFTypeRef refElementTop = CFDictionaryGetValue ((CFMutableDictionaryRef) refElement, CFSTR(kIOHIDElementKey)); + if (refElementTop) + { + CFTypeID type = CFGetTypeID (refElementTop); + if (type == CFArrayGetTypeID()) + { + CFRange range = {0, CFArrayGetCount ((CFArrayRef)refElementTop)}; + CFArrayApplyFunction ((CFArrayRef)refElementTop, range, getJoystickComponentArrayHandler, joyInfo); + } + } + } + } +} + +static void getJoystickComponentArrayHandler (const void * value, void * parameter) +{ + if (CFGetTypeID (value) == CFDictionaryGetTypeID ()) + addJoystickComponent ((CFTypeRef) value, (JoystickInfo *) parameter); +} + +static void joystickTopLevelElementHandler (const void * value, void * parameter) +{ + CFTypeRef refCF = 0; + if (CFGetTypeID (value) != CFDictionaryGetTypeID ()) + return; + refCF = CFDictionaryGetValue ((CFDictionaryRef)value, CFSTR(kIOHIDElementUsagePageKey)); + if (!CFNumberGetValue ((CFNumberRef)refCF, kCFNumberLongType, &((JoystickInfo *) parameter)->usagePage)) + irr::os::Printer::log("CFNumberGetValue error retrieving JoystickInfo->usagePage", irr::ELL_ERROR); + refCF = CFDictionaryGetValue ((CFDictionaryRef)value, CFSTR(kIOHIDElementUsageKey)); + if (!CFNumberGetValue ((CFNumberRef)refCF, kCFNumberLongType, &((JoystickInfo *) parameter)->usage)) + irr::os::Printer::log("CFNumberGetValue error retrieving JoystickInfo->usage", irr::ELL_ERROR); +} + +static void getJoystickDeviceInfo (io_object_t hidDevice, CFMutableDictionaryRef hidProperties, JoystickInfo *joyInfo) +{ + CFMutableDictionaryRef usbProperties = 0; + io_registry_entry_t parent1, parent2; + + /* Mac OS X currently is not mirroring all USB properties to HID page so need to look at USB device page also + * get dictionary for usb properties: step up two levels and get CF dictionary for USB properties + */ + if ((KERN_SUCCESS == IORegistryEntryGetParentEntry (hidDevice, kIOServicePlane, &parent1)) && + (KERN_SUCCESS == IORegistryEntryGetParentEntry (parent1, kIOServicePlane, &parent2)) && + (KERN_SUCCESS == IORegistryEntryCreateCFProperties (parent2, &usbProperties, kCFAllocatorDefault, kNilOptions))) + { + if (usbProperties) + { + CFTypeRef refCF = 0; + /* get device info + * try hid dictionary first, if fail then go to usb dictionary + */ + + /* get joystickName name */ + refCF = CFDictionaryGetValue (hidProperties, CFSTR(kIOHIDProductKey)); + if (!refCF) + refCF = CFDictionaryGetValue (usbProperties, CFSTR("USB Product Name")); + if (refCF) + { + if (!CFStringGetCString ((CFStringRef)refCF, joyInfo->joystickName, 256, CFStringGetSystemEncoding ())) + irr::os::Printer::log("CFStringGetCString error getting joyInfo->joystickName", irr::ELL_ERROR); + } + + /* get usage page and usage */ + refCF = CFDictionaryGetValue (hidProperties, CFSTR(kIOHIDPrimaryUsagePageKey)); + if (refCF) + { + if (!CFNumberGetValue ((CFNumberRef)refCF, kCFNumberLongType, &joyInfo->usagePage)) + irr::os::Printer::log("CFNumberGetValue error getting joyInfo->usagePage", irr::ELL_ERROR); + refCF = CFDictionaryGetValue (hidProperties, CFSTR(kIOHIDPrimaryUsageKey)); + if (refCF) + if (!CFNumberGetValue ((CFNumberRef)refCF, kCFNumberLongType, &joyInfo->usage)) + irr::os::Printer::log("CFNumberGetValue error getting joyInfo->usage", irr::ELL_ERROR); + } + + if (NULL == refCF) /* get top level element HID usage page or usage */ + { + /* use top level element instead */ + CFTypeRef refCFTopElement = 0; + refCFTopElement = CFDictionaryGetValue (hidProperties, CFSTR(kIOHIDElementKey)); + { + /* refCFTopElement points to an array of element dictionaries */ + CFRange range = {0, CFArrayGetCount ((CFArrayRef)refCFTopElement)}; + CFArrayApplyFunction ((CFArrayRef)refCFTopElement, range, joystickTopLevelElementHandler, joyInfo); + } + } + + CFRelease (usbProperties); + } + else + irr::os::Printer::log("IORegistryEntryCreateCFProperties failed to create usbProperties", irr::ELL_ERROR); + + if (kIOReturnSuccess != IOObjectRelease (parent2)) + irr::os::Printer::log("IOObjectRelease failed to release parent2", irr::ELL_ERROR); + if (kIOReturnSuccess != IOObjectRelease (parent1)) + irr::os::Printer::log("IOObjectRelease failed to release parent1", irr::ELL_ERROR); + } +} + +#endif // _IRR_COMPILE_WITH_JOYSTICK_EVENTS_ + +//------------------------------------------------------------------------------------------ +Boolean GetDictionaryBoolean(CFDictionaryRef theDict, const void* key) +{ + // get a boolean from the dictionary + Boolean value = false; + CFBooleanRef boolRef; + boolRef = (CFBooleanRef)CFDictionaryGetValue(theDict, key); + if (boolRef != NULL) + value = CFBooleanGetValue(boolRef); + return value; +} +//------------------------------------------------------------------------------------------ +long GetDictionaryLong(CFDictionaryRef theDict, const void* key) +{ + // get a long from the dictionary + long value = 0; + CFNumberRef numRef; + numRef = (CFNumberRef)CFDictionaryGetValue(theDict, key); + if (numRef != NULL) + CFNumberGetValue(numRef, kCFNumberLongType, &value); + return value; +} + +namespace irr +{ + namespace video + { + IVideoDriver* createOpenGLDriver(const SIrrlichtCreationParameters& param, io::IFileSystem* io, CIrrDeviceMacOSX *device); + } +} // end namespace irr + +static bool firstLaunch = true; + +namespace irr +{ +//! constructor +CIrrDeviceMacOSX::CIrrDeviceMacOSX(const SIrrlichtCreationParameters& param) + : CIrrDeviceStub(param), Window(NULL), CGLContext(NULL), OGLContext(NULL), + SoftwareDriverTarget(0), DeviceWidth(0), DeviceHeight(0), + ScreenWidth(0), ScreenHeight(0), MouseButtonStates(0), SoftwareRendererType(0), + IsActive(true), IsFullscreen(false), IsShiftDown(false), IsControlDown(false), IsResizable(false) +{ + struct utsname name; + NSString *path; + + #ifdef _DEBUG + setDebugName("CIrrDeviceMacOSX"); + #endif + + if (firstLaunch) + { + firstLaunch = false; + + if(!CreationParams.WindowId) //load menus if standalone application + { + [[NSAutoreleasePool alloc] init]; + [NSApplication sharedApplication]; + [NSApp setDelegate:[[[AppDelegate alloc] initWithDevice:this] autorelease]]; + [NSBundle loadNibNamed:@"MainMenu" owner:[NSApp delegate]]; + [NSApp finishLaunching]; + } + + path = [[[NSBundle mainBundle] bundlePath] stringByDeletingLastPathComponent]; + chdir([path fileSystemRepresentation]); + [path release]; + } + + uname(&name); + Operator = new COSOperator(name.version); + os::Printer::log(name.version,ELL_INFORMATION); + + initKeycodes(); + + VideoModeList.setDesktop(CreationParams.Bits, core::dimension2d([[NSScreen mainScreen] frame].size.width, [[NSScreen mainScreen] frame].size.height)); + + bool success = true; + if (CreationParams.DriverType != video::EDT_NULL) + success = createWindow(); + + // in case of failure, one can check VideoDriver for initialization + if (!success) + return; + + setResizable(false); + CursorControl = new CCursorControl(CreationParams.WindowSize, this); + + createDriver(); + createGUIAndScene(); +} + +CIrrDeviceMacOSX::~CIrrDeviceMacOSX() +{ + [SoftwareDriverTarget release]; +#ifdef __MAC_10_6 + [NSApp setPresentationOptions:(NSApplicationPresentationDefault)]; +#else + SetSystemUIMode(kUIModeNormal, kUIOptionAutoShowMenuBar); +#endif + closeDevice(); +#if defined(_IRR_COMPILE_WITH_JOYSTICK_EVENTS_) + for (u32 joystick = 0; joystick < ActiveJoysticks.size(); ++joystick) + { + if (ActiveJoysticks[joystick].interface) + closeJoystickDevice(&ActiveJoysticks[joystick]); + } +#endif +} + +void CIrrDeviceMacOSX::closeDevice() +{ + if (Window != NULL) + { + [Window setIsVisible:FALSE]; + + if (OGLContext != NULL) + { + [OGLContext clearDrawable]; + [OGLContext release]; + OGLContext = NULL; + } + + [Window setReleasedWhenClosed:TRUE]; + [Window release]; + Window = NULL; + + if (IsFullscreen) + CGReleaseAllDisplays(); + } + else + { + if (CGLContext != NULL) + { + if(CreationParams.WindowId) + { + [(NSOpenGLContext *)OGLContext clearDrawable]; + [(NSOpenGLContext *)OGLContext release]; + OGLContext = NULL; + } + else + { + CGLSetCurrentContext(NULL); + CGLClearDrawable(CGLContext); + CGLDestroyContext(CGLContext); + CGReleaseAllDisplays(); + } + } + } + + IsFullscreen = false; + IsActive = false; + CGLContext = NULL; +} + +bool CIrrDeviceMacOSX::createWindow() +{ + CGDisplayErr error; + bool result=false; + CGDirectDisplayID display=CGMainDisplayID(); + CGLPixelFormatObj pixelFormat; + CGRect displayRect; +#ifdef __MAC_10_6 + CGDisplayModeRef displaymode, olddisplaymode; +#else + CFDictionaryRef displaymode, olddisplaymode; +#endif + GLint numPixelFormats, newSwapInterval; + + int alphaSize = CreationParams.WithAlphaChannel?4:0; + int depthSize = CreationParams.ZBufferBits; + if (CreationParams.WithAlphaChannel && (CreationParams.Bits == 32)) + alphaSize = 8; + + ScreenWidth = (int) CGDisplayPixelsWide(display); + ScreenHeight = (int) CGDisplayPixelsHigh(display); + + // we need to check where the exceptions may happen and work at them + // for now we will just catch them to be able to avoid an app exit + @try + { + if (!CreationParams.Fullscreen) + { + if(!CreationParams.WindowId) //create another window when WindowId is null + { + NSBackingStoreType type = (CreationParams.DriverType == video::EDT_OPENGL) ? NSBackingStoreBuffered : NSBackingStoreNonretained; + + Window = [[NSWindow alloc] initWithContentRect:NSMakeRect(0,0,CreationParams.WindowSize.Width,CreationParams.WindowSize.Height) styleMask:NSTitledWindowMask+NSClosableWindowMask+NSResizableWindowMask backing:type defer:FALSE]; + } + + if (Window != NULL || CreationParams.WindowId) + { + if (CreationParams.DriverType == video::EDT_OPENGL) + { + NSOpenGLPixelFormatAttribute windowattribs[] = + { + NSOpenGLPFANoRecovery, + NSOpenGLPFAAccelerated, + NSOpenGLPFADepthSize, (NSOpenGLPixelFormatAttribute)depthSize, + NSOpenGLPFAColorSize, (NSOpenGLPixelFormatAttribute)CreationParams.Bits, + NSOpenGLPFAAlphaSize, (NSOpenGLPixelFormatAttribute)alphaSize, + NSOpenGLPFASampleBuffers, (NSOpenGLPixelFormatAttribute)1, + NSOpenGLPFASamples, (NSOpenGLPixelFormatAttribute)CreationParams.AntiAlias, + NSOpenGLPFAStencilSize, (NSOpenGLPixelFormatAttribute)(CreationParams.Stencilbuffer?1:0), + NSOpenGLPFADoubleBuffer, + (NSOpenGLPixelFormatAttribute)nil + }; + + if (CreationParams.AntiAlias<2) + { + windowattribs[ 9] = (NSOpenGLPixelFormatAttribute)0; + windowattribs[11] = (NSOpenGLPixelFormatAttribute)0; + } + + NSOpenGLPixelFormat *format; + for (int i=0; i<3; ++i) + { + if (1==i) + { + // Second try without stencilbuffer + if (CreationParams.Stencilbuffer) + { + windowattribs[13]=(NSOpenGLPixelFormatAttribute)0; + } + else + continue; + } + else if (2==i) + { + // Third try without Doublebuffer + os::Printer::log("No doublebuffering available.", ELL_WARNING); + windowattribs[14]=(NSOpenGLPixelFormatAttribute)nil; + } + + format = [[NSOpenGLPixelFormat alloc] initWithAttributes:windowattribs]; + if (format == NULL) + { + if (CreationParams.AntiAlias>1) + { + while (!format && windowattribs[12]>1) + { + windowattribs[12] = (NSOpenGLPixelFormatAttribute)((int)windowattribs[12]-1); + format = [[NSOpenGLPixelFormat alloc] initWithAttributes:windowattribs]; + } + + if (!format) + { + windowattribs[9] = (NSOpenGLPixelFormatAttribute)0; + windowattribs[11] = (NSOpenGLPixelFormatAttribute)0; + format = [[NSOpenGLPixelFormat alloc] initWithAttributes:windowattribs]; + if (!format) + { + // reset values for next try + windowattribs[9] = (NSOpenGLPixelFormatAttribute)1; + windowattribs[11] = (NSOpenGLPixelFormatAttribute)CreationParams.AntiAlias; + } + else + { + os::Printer::log("No FSAA available.", ELL_WARNING); + } + } + } + } + else + break; + } + CreationParams.AntiAlias = windowattribs[11]; + CreationParams.Stencilbuffer=(windowattribs[13]==1); + + if (format != NULL) + { + OGLContext = [[NSOpenGLContext alloc] initWithFormat:format shareContext:NULL]; + [format release]; + } + } + + if (OGLContext != NULL || CreationParams.DriverType != video::EDT_OPENGL) + { + if (!CreationParams.WindowId) + { + [Window center]; + [Window setDelegate:[NSApp delegate]]; + + if(CreationParams.DriverType == video::EDT_OPENGL) + [OGLContext setView:[Window contentView]]; + + [Window setAcceptsMouseMovedEvents:TRUE]; + [Window setIsVisible:TRUE]; + [Window makeKeyAndOrderFront:nil]; + } + else if(CreationParams.DriverType == video::EDT_OPENGL) //use another window for drawing + [OGLContext setView:(NSView*)CreationParams.WindowId]; + + if (CreationParams.DriverType == video::EDT_OPENGL) + CGLContext = (CGLContextObj) [OGLContext CGLContextObj]; + + DeviceWidth = CreationParams.WindowSize.Width; + DeviceHeight = CreationParams.WindowSize.Height; + result = true; + } + } + } + else + { + IsFullscreen = true; + +#ifdef __MAC_10_6 + displaymode = CGDisplayCopyDisplayMode(display); + + CFArrayRef Modes = CGDisplayCopyAllDisplayModes(display, NULL); + + for(int i = 0; i < CFArrayGetCount(Modes); ++i) + { + CGDisplayModeRef CurrentMode = (CGDisplayModeRef)CFArrayGetValueAtIndex(Modes, i); + + u8 Depth = 0; + + CFStringRef pixEnc = CGDisplayModeCopyPixelEncoding(CurrentMode); + + if(CFStringCompare(pixEnc, CFSTR(IO32BitDirectPixels), kCFCompareCaseInsensitive) == kCFCompareEqualTo) + Depth = 32; + else + if(CFStringCompare(pixEnc, CFSTR(IO16BitDirectPixels), kCFCompareCaseInsensitive) == kCFCompareEqualTo) + Depth = 16; + else + if(CFStringCompare(pixEnc, CFSTR(IO8BitIndexedPixels), kCFCompareCaseInsensitive) == kCFCompareEqualTo) + Depth = 8; + + if(Depth == CreationParams.Bits) + if((CGDisplayModeGetWidth(CurrentMode) == CreationParams.WindowSize.Width) && (CGDisplayModeGetHeight(CurrentMode) == CreationParams.WindowSize.Height)) + { + displaymode = CurrentMode; + break; + } + } +#else + displaymode = CGDisplayBestModeForParameters(display,CreationParams.Bits,CreationParams.WindowSize.Width,CreationParams.WindowSize.Height,NULL); +#endif + + if (displaymode != NULL) + { +#ifdef __MAC_10_6 + olddisplaymode = CGDisplayCopyDisplayMode(display); +#else + olddisplaymode = CGDisplayCurrentMode(display); +#endif + + error = CGCaptureAllDisplays(); + if (error == CGDisplayNoErr) + { +#ifdef __MAC_10_6 + error = CGDisplaySetDisplayMode(display, displaymode, NULL); +#else + error = CGDisplaySwitchToMode(display, displaymode); +#endif + + if (error == CGDisplayNoErr) + { + if (CreationParams.DriverType == video::EDT_OPENGL) + { + CGLPixelFormatAttribute fullattribs[] = + { + kCGLPFAFullScreen, + kCGLPFADisplayMask, (CGLPixelFormatAttribute)CGDisplayIDToOpenGLDisplayMask(display), + kCGLPFADoubleBuffer, + kCGLPFANoRecovery, + kCGLPFAAccelerated, + kCGLPFADepthSize, (CGLPixelFormatAttribute)depthSize, + kCGLPFAColorSize, (CGLPixelFormatAttribute)CreationParams.Bits, + kCGLPFAAlphaSize, (CGLPixelFormatAttribute)alphaSize, + kCGLPFASampleBuffers, (CGLPixelFormatAttribute)(CreationParams.AntiAlias?1:0), + kCGLPFASamples, (CGLPixelFormatAttribute)CreationParams.AntiAlias, + kCGLPFAStencilSize, (CGLPixelFormatAttribute)(CreationParams.Stencilbuffer?1:0), + (CGLPixelFormatAttribute)NULL + }; + + pixelFormat = NULL; + numPixelFormats = 0; + CGLChoosePixelFormat(fullattribs,&pixelFormat,&numPixelFormats); + + if (pixelFormat != NULL) + { + CGLCreateContext(pixelFormat,NULL,&CGLContext); + CGLDestroyPixelFormat(pixelFormat); + } + + if (CGLContext != NULL) + { +#ifdef __MAC_10_6 + CGLSetFullScreenOnDisplay(CGLContext, CGDisplayIDToOpenGLDisplayMask(display)); +#else + CGLSetFullScreen(CGLContext); +#endif + displayRect = CGDisplayBounds(display); + ScreenWidth = DeviceWidth = (int)displayRect.size.width; + ScreenHeight = DeviceHeight = (int)displayRect.size.height; + CreationParams.WindowSize.set(ScreenWidth, ScreenHeight); + result = true; + } + } + else + { + Window = [[NSWindow alloc] initWithContentRect:[[NSScreen mainScreen] frame] styleMask:NSBorderlessWindowMask backing:NSBackingStoreNonretained defer:NO screen:[NSScreen mainScreen]]; + + [Window setLevel: CGShieldingWindowLevel()]; + [Window setAcceptsMouseMovedEvents:TRUE]; + [Window setIsVisible:TRUE]; + [Window makeKeyAndOrderFront:nil]; + + displayRect = CGDisplayBounds(display); + ScreenWidth = DeviceWidth = (int)displayRect.size.width; + ScreenHeight = DeviceHeight = (int)displayRect.size.height; + CreationParams.WindowSize.set(ScreenWidth, ScreenHeight); + result = true; + } + } + if (!result) + CGReleaseAllDisplays(); + } + } + } + } + @catch (NSException *exception) + { + closeDevice(); + result = false; + } + + if (result) + { + // fullscreen? + if (Window == NULL && !CreationParams.WindowId) //hide menus in fullscreen mode only +#ifdef __MAC_10_6 + [NSApp setPresentationOptions:(NSApplicationPresentationAutoHideDock | NSApplicationPresentationAutoHideMenuBar)]; +#else + SetSystemUIMode(kUIModeAllHidden, kUIOptionAutoShowMenuBar); +#endif + + if(CreationParams.DriverType == video::EDT_OPENGL) + { + CGLSetCurrentContext(CGLContext); + newSwapInterval = (CreationParams.Vsync) ? 1 : 0; + CGLSetParameter(CGLContext,kCGLCPSwapInterval,&newSwapInterval); + } + } + + return (result); +} + +void CIrrDeviceMacOSX::setResize(int width, int height) +{ + // set new window size + DeviceWidth = width; + DeviceHeight = height; + + // update the size of the opengl rendering context + if(OGLContext); + [OGLContext update]; + + // resize the driver to the inner pane size + if (Window) + { + NSRect driverFrame = [Window contentRectForFrameRect:[Window frame]]; + getVideoDriver()->OnResize(core::dimension2d( (s32)driverFrame.size.width, (s32)driverFrame.size.height)); + } + else + getVideoDriver()->OnResize(core::dimension2d( (s32)width, (s32)height)); + + if (CreationParams.WindowId && OGLContext) + [(NSOpenGLContext *)OGLContext update]; +} + + +void CIrrDeviceMacOSX::createDriver() +{ + switch (CreationParams.DriverType) + { + case video::EDT_SOFTWARE: + #ifdef _IRR_COMPILE_WITH_SOFTWARE_ + VideoDriver = video::createSoftwareDriver(CreationParams.WindowSize, CreationParams.Fullscreen, FileSystem, this); + SoftwareRendererType = 2; + #else + os::Printer::log("No Software driver support compiled in.", ELL_ERROR); + #endif + break; + + case video::EDT_BURNINGSVIDEO: + #ifdef _IRR_COMPILE_WITH_BURNINGSVIDEO_ + VideoDriver = video::createBurningVideoDriver(CreationParams, FileSystem, this); + SoftwareRendererType = 1; + #else + os::Printer::log("Burning's video driver was not compiled in.", ELL_ERROR); + #endif + break; + + case video::EDT_OPENGL: + #ifdef _IRR_COMPILE_WITH_OPENGL_ + VideoDriver = video::createOpenGLDriver(CreationParams, FileSystem, this); + #else + os::Printer::log("No OpenGL support compiled in.", ELL_ERROR); + #endif + break; + + case video::EDT_DIRECT3D8: + case video::EDT_DIRECT3D9: + os::Printer::log("This driver is not available in OSX. Try OpenGL or Software renderer.", ELL_ERROR); + break; + + case video::EDT_NULL: + VideoDriver = video::createNullDriver(FileSystem, CreationParams.WindowSize); + break; + + default: + os::Printer::log("Unable to create video driver of unknown type.", ELL_ERROR); + break; + } +} + +void CIrrDeviceMacOSX::flush() +{ + if (CGLContext != NULL) + { + glFinish(); + CGLFlushDrawable(CGLContext); + } +} + +bool CIrrDeviceMacOSX::run() +{ + NSAutoreleasePool* Pool = [[NSAutoreleasePool alloc] init]; + + NSEvent *event; + irr::SEvent ievent; + + os::Timer::tick(); + storeMouseLocation(); + + event = [NSApp nextEventMatchingMask:NSAnyEventMask untilDate:[NSDate distantPast] inMode:NSDefaultRunLoopMode dequeue:YES]; + if (event != nil) + { + bzero(&ievent,sizeof(ievent)); + + switch([(NSEvent *)event type]) + { + case NSKeyDown: + postKeyEvent(event,ievent,true); + break; + + case NSKeyUp: + postKeyEvent(event,ievent,false); + break; + + case NSFlagsChanged: + ievent.EventType = irr::EET_KEY_INPUT_EVENT; + ievent.KeyInput.Shift = ([(NSEvent *)event modifierFlags] & NSShiftKeyMask) != 0; + ievent.KeyInput.Control = ([(NSEvent *)event modifierFlags] & NSControlKeyMask) != 0; + + if (IsShiftDown != ievent.KeyInput.Shift) + { + ievent.KeyInput.Char = irr::KEY_SHIFT; + ievent.KeyInput.Key = irr::KEY_SHIFT; + ievent.KeyInput.PressedDown = ievent.KeyInput.Shift; + + IsShiftDown = ievent.KeyInput.Shift; + + postEventFromUser(ievent); + } + + if (IsControlDown != ievent.KeyInput.Control) + { + ievent.KeyInput.Char = irr::KEY_CONTROL; + ievent.KeyInput.Key = irr::KEY_CONTROL; + ievent.KeyInput.PressedDown = ievent.KeyInput.Control; + + IsControlDown = ievent.KeyInput.Control; + + postEventFromUser(ievent); + } + + [NSApp sendEvent:event]; + break; + + case NSLeftMouseDown: + ievent.EventType = irr::EET_MOUSE_INPUT_EVENT; + ievent.MouseInput.Event = irr::EMIE_LMOUSE_PRESSED_DOWN; + MouseButtonStates |= irr::EMBSM_LEFT; + ievent.MouseInput.ButtonStates = MouseButtonStates; + postMouseEvent(event,ievent); + break; + + case NSLeftMouseUp: + ievent.EventType = irr::EET_MOUSE_INPUT_EVENT; + MouseButtonStates &= !irr::EMBSM_LEFT; + ievent.MouseInput.ButtonStates = MouseButtonStates; + ievent.MouseInput.Event = irr::EMIE_LMOUSE_LEFT_UP; + postMouseEvent(event,ievent); + break; + + case NSOtherMouseDown: + ievent.EventType = irr::EET_MOUSE_INPUT_EVENT; + ievent.MouseInput.Event = irr::EMIE_MMOUSE_PRESSED_DOWN; + MouseButtonStates |= irr::EMBSM_MIDDLE; + ievent.MouseInput.ButtonStates = MouseButtonStates; + postMouseEvent(event,ievent); + break; + + case NSOtherMouseUp: + ievent.EventType = irr::EET_MOUSE_INPUT_EVENT; + MouseButtonStates &= !irr::EMBSM_MIDDLE; + ievent.MouseInput.ButtonStates = MouseButtonStates; + ievent.MouseInput.Event = irr::EMIE_MMOUSE_LEFT_UP; + postMouseEvent(event,ievent); + break; + + case NSMouseMoved: + case NSLeftMouseDragged: + case NSRightMouseDragged: + case NSOtherMouseDragged: + ievent.EventType = irr::EET_MOUSE_INPUT_EVENT; + ievent.MouseInput.Event = irr::EMIE_MOUSE_MOVED; + ievent.MouseInput.ButtonStates = MouseButtonStates; + postMouseEvent(event,ievent); + break; + + case NSRightMouseDown: + ievent.EventType = irr::EET_MOUSE_INPUT_EVENT; + ievent.MouseInput.Event = irr::EMIE_RMOUSE_PRESSED_DOWN; + MouseButtonStates |= irr::EMBSM_RIGHT; + ievent.MouseInput.ButtonStates = MouseButtonStates; + postMouseEvent(event,ievent); + break; + + case NSRightMouseUp: + ievent.EventType = irr::EET_MOUSE_INPUT_EVENT; + ievent.MouseInput.Event = irr::EMIE_RMOUSE_LEFT_UP; + MouseButtonStates &= !irr::EMBSM_RIGHT; + ievent.MouseInput.ButtonStates = MouseButtonStates; + postMouseEvent(event,ievent); + break; + + case NSScrollWheel: + ievent.EventType = irr::EET_MOUSE_INPUT_EVENT; + ievent.MouseInput.Event = irr::EMIE_MOUSE_WHEEL; + ievent.MouseInput.Wheel = [(NSEvent *)event deltaY]; + if (ievent.MouseInput.Wheel < 1.0f) + ievent.MouseInput.Wheel *= 10.0f; + else + ievent.MouseInput.Wheel *= 5.0f; + postMouseEvent(event,ievent); + break; + + default: + [NSApp sendEvent:event]; + break; + } + } + + pollJoysticks(); + + [Pool release]; + + return (![[NSApp delegate] isQuit] && IsActive); +} + + +//! Pause the current process for the minimum time allowed only to allow other processes to execute +void CIrrDeviceMacOSX::yield() +{ + struct timespec ts = {0,0}; + nanosleep(&ts, NULL); +} + + +//! Pause execution and let other processes to run for a specified amount of time. +void CIrrDeviceMacOSX::sleep(u32 timeMs, bool pauseTimer=false) +{ + bool wasStopped = Timer ? Timer->isStopped() : true; + + struct timespec ts; + ts.tv_sec = (time_t) (timeMs / 1000); + ts.tv_nsec = (long) (timeMs % 1000) * 1000000; + + if (pauseTimer && !wasStopped) + Timer->stop(); + + nanosleep(&ts, NULL); + + if (pauseTimer && !wasStopped) + Timer->start(); +} + + +void CIrrDeviceMacOSX::setWindowCaption(const wchar_t* text) +{ + size_t size; + char title[1024]; + + if (Window != NULL) + { + size = wcstombs(title,text,1024); + title[1023] = 0; +#ifdef __MAC_10_6 + NSString* name = [NSString stringWithCString:title encoding:NSUTF8StringEncoding]; +#else + NSString* name = [NSString stringWithCString:title length:size]; +#endif + [Window setTitle:name]; + [name release]; + } +} + + +bool CIrrDeviceMacOSX::isWindowActive() const +{ + return (IsActive); +} + + +bool CIrrDeviceMacOSX::isWindowFocused() const +{ + if (Window != NULL) + return [Window isKeyWindow]; + return false; +} + + +bool CIrrDeviceMacOSX::isWindowMinimized() const +{ + if (Window != NULL) + return [Window isMiniaturized]; + return false; +} + + +void CIrrDeviceMacOSX::postKeyEvent(void *event,irr::SEvent &ievent,bool pressed) +{ + NSString *str; + std::map::const_iterator iter; + unsigned int result,c,mkey,mchar; + const unsigned char *cStr; + BOOL skipCommand; + + str = [(NSEvent *)event characters]; + if ((str != nil) && ([str length] > 0)) + { + mkey = mchar = 0; + skipCommand = false; + c = [str characterAtIndex:0]; + mchar = c; + + iter = KeyCodes.find([(NSEvent *)event keyCode]); + if (iter != KeyCodes.end()) + mkey = (*iter).second; + else if ((iter = KeyCodes.find(c)) != KeyCodes.end()) + mkey = (*iter).second; + else + { + // workaround for period character + if (c == 0x2E) + { + mkey = irr::KEY_PERIOD; + mchar = '.'; + } + else + { + cStr = (unsigned char *)[str cStringUsingEncoding:NSWindowsCP1252StringEncoding]; + if (cStr != NULL && strlen((char*)cStr) > 0) + { + mchar = cStr[0]; + mkey = toupper(mchar); + if ([(NSEvent *)event modifierFlags] & NSCommandKeyMask) + { + if (mkey == 'C' || mkey == 'V' || mkey == 'X') + { + mchar = 0; + skipCommand = true; + } + } + } + } + } + + ievent.EventType = irr::EET_KEY_INPUT_EVENT; + ievent.KeyInput.Key = (irr::EKEY_CODE)mkey; + ievent.KeyInput.PressedDown = pressed; + ievent.KeyInput.Shift = ([(NSEvent *)event modifierFlags] & NSShiftKeyMask) != 0; + ievent.KeyInput.Control = ([(NSEvent *)event modifierFlags] & NSControlKeyMask) != 0; + ievent.KeyInput.Char = mchar; + + if (skipCommand) + ievent.KeyInput.Control = true; + else if ([(NSEvent *)event modifierFlags] & NSCommandKeyMask) + [NSApp sendEvent:(NSEvent *)event]; + + postEventFromUser(ievent); + } +} + + +void CIrrDeviceMacOSX::postMouseEvent(void *event,irr::SEvent &ievent) +{ + bool post = true; + + if (Window != NULL) + { + ievent.MouseInput.X = (int)[(NSEvent *)event locationInWindow].x; + ievent.MouseInput.Y = DeviceHeight - (int)[(NSEvent *)event locationInWindow].y; + + if (ievent.MouseInput.Y < 0) + post = false; + } + else + { + CGEventRef ourEvent = CGEventCreate(NULL); + CGPoint point = CGEventGetLocation(ourEvent); + CFRelease(ourEvent); + + ievent.MouseInput.X = (int)point.x; + ievent.MouseInput.Y = (int)point.y; + + if (ievent.MouseInput.Y < 0) + post = false; + } + + if (post) + postEventFromUser(ievent); + + [NSApp sendEvent:(NSEvent *)event]; +} + + +void CIrrDeviceMacOSX::storeMouseLocation() +{ + int x,y; + + if (Window != NULL) + { + NSPoint p; + p = [NSEvent mouseLocation]; + p = [Window convertScreenToBase:p]; + x = (int)p.x; + y = DeviceHeight - (int)p.y; + } + else + { + CGEventRef ourEvent = CGEventCreate(NULL); + CGPoint point = CGEventGetLocation(ourEvent); + CFRelease(ourEvent); + + x = (int)point.x; + y = (int)point.y; + + const core::position2di& curr = ((CCursorControl *)CursorControl)->getPosition(); + if (curr.X != x || curr.Y != y) + { + // In fullscreen mode, events are not sent regularly so rely on polling + irr::SEvent ievent; + ievent.EventType = irr::EET_MOUSE_INPUT_EVENT; + ievent.MouseInput.Event = irr::EMIE_MOUSE_MOVED; + ievent.MouseInput.X = x; + ievent.MouseInput.Y = y; + postEventFromUser(ievent); + } + } + + ((CCursorControl *)CursorControl)->updateInternalCursorPosition(x,y); +} + + +void CIrrDeviceMacOSX::setMouseLocation(int x,int y) +{ + NSPoint p; + CGPoint c; + + if (Window != NULL) + { + // Irrlicht window exists + p.x = (float) x; + p.y = (float) (DeviceHeight - y); + p = [Window convertBaseToScreen:p]; + p.y = ScreenHeight - p.y; + } + else + { + p.x = (float) x; + p.y = (float) y + (ScreenHeight - DeviceHeight); + } + + c.x = p.x; + c.y = p.y; + +#ifdef __MAC_10_6 + /*CGEventSourceRef SourceRef = CGEventSourceCreate(0); + CGEventSourceSetLocalEventsSuppressionInterval(SourceRef, 0); + CFRelease(SourceRef);*/ + CGSetLocalEventsSuppressionInterval(0); +#else + CGSetLocalEventsSuppressionInterval(0); +#endif + CGWarpMouseCursorPosition(c); +} + + +void CIrrDeviceMacOSX::setCursorVisible(bool visible) +{ + if (visible) + CGDisplayShowCursor(CGMainDisplayID()); + else + CGDisplayHideCursor(CGMainDisplayID()); +} + + +void CIrrDeviceMacOSX::initKeycodes() +{ + KeyCodes[kVK_UpArrow] = irr::KEY_UP; + KeyCodes[kVK_DownArrow] = irr::KEY_DOWN; + KeyCodes[kVK_LeftArrow] = irr::KEY_LEFT; + KeyCodes[kVK_RightArrow] = irr::KEY_RIGHT; + KeyCodes[kVK_F1] = irr::KEY_F1; + KeyCodes[kVK_F2] = irr::KEY_F2; + KeyCodes[kVK_F3] = irr::KEY_F3; + KeyCodes[kVK_F4] = irr::KEY_F4; + KeyCodes[kVK_F5] = irr::KEY_F5; + KeyCodes[kVK_F6] = irr::KEY_F6; + KeyCodes[kVK_F7] = irr::KEY_F7; + KeyCodes[kVK_F8] = irr::KEY_F8; + KeyCodes[kVK_F9] = irr::KEY_F9; + KeyCodes[kVK_F10] = irr::KEY_F10; + KeyCodes[kVK_F11] = irr::KEY_F11; + KeyCodes[kVK_F12] = irr::KEY_F12; + KeyCodes[kVK_F13] = irr::KEY_F13; + KeyCodes[kVK_F14] = irr::KEY_F14; + KeyCodes[kVK_F15] = irr::KEY_F15; + KeyCodes[kVK_F16] = irr::KEY_F16; + KeyCodes[kVK_F17] = irr::KEY_F17; + KeyCodes[kVK_F18] = irr::KEY_F18; + KeyCodes[kVK_F19] = irr::KEY_F19; + KeyCodes[kVK_F20] = irr::KEY_F20; + KeyCodes[kVK_Home] = irr::KEY_HOME; + KeyCodes[kVK_End] = irr::KEY_END; + KeyCodes[NSInsertFunctionKey] = irr::KEY_INSERT; + KeyCodes[kVK_ForwardDelete] = irr::KEY_DELETE; + KeyCodes[kVK_Help] = irr::KEY_HELP; + KeyCodes[NSSelectFunctionKey] = irr::KEY_SELECT; + KeyCodes[NSPrintFunctionKey] = irr::KEY_PRINT; + KeyCodes[NSExecuteFunctionKey] = irr::KEY_EXECUT; + KeyCodes[NSPrintScreenFunctionKey] = irr::KEY_SNAPSHOT; + KeyCodes[NSPauseFunctionKey] = irr::KEY_PAUSE; + KeyCodes[NSScrollLockFunctionKey] = irr::KEY_SCROLL; + KeyCodes[kVK_Delete] = irr::KEY_BACK; + KeyCodes[kVK_Tab] = irr::KEY_TAB; + KeyCodes[kVK_Return] = irr::KEY_RETURN; + KeyCodes[kVK_Escape] = irr::KEY_ESCAPE; + KeyCodes[kVK_Control] = irr::KEY_CONTROL; + KeyCodes[kVK_RightControl] = irr::KEY_RCONTROL; + KeyCodes[kVK_Command] = irr::KEY_MENU; + KeyCodes[kVK_Shift] = irr::KEY_SHIFT; + KeyCodes[kVK_RightShift] = irr::KEY_RSHIFT; + KeyCodes[kVK_Space] = irr::KEY_SPACE; + + KeyCodes[kVK_ANSI_A] = irr::KEY_KEY_A; + KeyCodes[kVK_ANSI_B] = irr::KEY_KEY_B; + KeyCodes[kVK_ANSI_C] = irr::KEY_KEY_C; + KeyCodes[kVK_ANSI_D] = irr::KEY_KEY_D; + KeyCodes[kVK_ANSI_E] = irr::KEY_KEY_E; + KeyCodes[kVK_ANSI_F] = irr::KEY_KEY_F; + KeyCodes[kVK_ANSI_G] = irr::KEY_KEY_G; + KeyCodes[kVK_ANSI_H] = irr::KEY_KEY_H; + KeyCodes[kVK_ANSI_I] = irr::KEY_KEY_I; + KeyCodes[kVK_ANSI_J] = irr::KEY_KEY_J; + KeyCodes[kVK_ANSI_K] = irr::KEY_KEY_K; + KeyCodes[kVK_ANSI_L] = irr::KEY_KEY_L; + KeyCodes[kVK_ANSI_M] = irr::KEY_KEY_M; + KeyCodes[kVK_ANSI_N] = irr::KEY_KEY_N; + KeyCodes[kVK_ANSI_O] = irr::KEY_KEY_O; + KeyCodes[kVK_ANSI_P] = irr::KEY_KEY_P; + KeyCodes[kVK_ANSI_Q] = irr::KEY_KEY_Q; + KeyCodes[kVK_ANSI_R] = irr::KEY_KEY_R; + KeyCodes[kVK_ANSI_S] = irr::KEY_KEY_S; + KeyCodes[kVK_ANSI_T] = irr::KEY_KEY_T; + KeyCodes[kVK_ANSI_U] = irr::KEY_KEY_U; + KeyCodes[kVK_ANSI_V] = irr::KEY_KEY_V; + KeyCodes[kVK_ANSI_W] = irr::KEY_KEY_W; + KeyCodes[kVK_ANSI_X] = irr::KEY_KEY_X; + KeyCodes[kVK_ANSI_X] = irr::KEY_KEY_X; + KeyCodes[kVK_ANSI_Y] = irr::KEY_KEY_Y; + KeyCodes[kVK_ANSI_Z] = irr::KEY_KEY_Z; + + KeyCodes[kVK_ANSI_0] = irr::KEY_KEY_0; + KeyCodes[kVK_ANSI_1] = irr::KEY_KEY_1; + KeyCodes[kVK_ANSI_2] = irr::KEY_KEY_2; + KeyCodes[kVK_ANSI_3] = irr::KEY_KEY_3; + KeyCodes[kVK_ANSI_4] = irr::KEY_KEY_4; + KeyCodes[kVK_ANSI_5] = irr::KEY_KEY_5; + KeyCodes[kVK_ANSI_6] = irr::KEY_KEY_6; + KeyCodes[kVK_ANSI_7] = irr::KEY_KEY_7; + KeyCodes[kVK_ANSI_8] = irr::KEY_KEY_8; + KeyCodes[kVK_ANSI_9] = irr::KEY_KEY_9; + + KeyCodes[kVK_ANSI_Slash] = irr::KEY_DIVIDE; + KeyCodes[kVK_ANSI_Comma] = irr::KEY_COMMA; + KeyCodes[kVK_ANSI_Period] = irr::KEY_PERIOD; + KeyCodes[kVK_PageUp] = irr::KEY_PRIOR; + KeyCodes[kVK_PageDown] = irr::KEY_NEXT; + + KeyCodes[kVK_ANSI_Keypad0] = irr::KEY_NUMPAD0; + KeyCodes[kVK_ANSI_Keypad1] = irr::KEY_NUMPAD1; + KeyCodes[kVK_ANSI_Keypad2] = irr::KEY_NUMPAD2; + KeyCodes[kVK_ANSI_Keypad3] = irr::KEY_NUMPAD3; + KeyCodes[kVK_ANSI_Keypad4] = irr::KEY_NUMPAD4; + KeyCodes[kVK_ANSI_Keypad5] = irr::KEY_NUMPAD5; + KeyCodes[kVK_ANSI_Keypad6] = irr::KEY_NUMPAD6; + KeyCodes[kVK_ANSI_Keypad7] = irr::KEY_NUMPAD7; + KeyCodes[kVK_ANSI_Keypad8] = irr::KEY_NUMPAD8; + KeyCodes[kVK_ANSI_Keypad9] = irr::KEY_NUMPAD9; + + KeyCodes[kVK_ANSI_KeypadDecimal] = irr::KEY_DECIMAL; + KeyCodes[kVK_ANSI_KeypadMultiply] = irr::KEY_MULTIPLY; + KeyCodes[kVK_ANSI_KeypadPlus] = irr::KEY_PLUS; + KeyCodes[kVK_ANSI_KeypadClear] = irr::KEY_OEM_CLEAR; + KeyCodes[kVK_ANSI_KeypadDivide] = irr::KEY_DIVIDE; + KeyCodes[kVK_ANSI_KeypadEnter] = irr::KEY_RETURN; + KeyCodes[kVK_ANSI_KeypadMinus] = irr::KEY_SUBTRACT; +} + + +//! Sets if the window should be resizable in windowed mode. +void CIrrDeviceMacOSX::setResizable(bool resize) +{ + IsResizable = resize; +#if 0 + if (resize) + [Window setStyleMask:NSTitledWindowMask|NSClosableWindowMask|NSMiniaturizableWindowMask|NSResizableWindowMask]; + else + [Window setStyleMask:NSTitledWindowMask|NSClosableWindowMask]; +#endif +} + + +bool CIrrDeviceMacOSX::isResizable() const +{ + return IsResizable; +} + + +void CIrrDeviceMacOSX::minimizeWindow() +{ + if (Window != NULL) + [Window miniaturize:[NSApp self]]; +} + + +//! Maximizes the window if possible. +void CIrrDeviceMacOSX::maximizeWindow() +{ + // todo: implement +} + + +//! Restore the window to normal size if possible. +void CIrrDeviceMacOSX::restoreWindow() +{ + [Window deminiaturize:[NSApp self]]; +} + + +bool CIrrDeviceMacOSX::present(video::IImage* surface, void* windowId, core::rect* src ) +{ + // todo: implement window ID and src rectangle + + if (!surface) + return false; + + if (SoftwareRendererType > 0) + { + const u32 colorSamples=3; + // do we need to change the size? + const bool updateSize = !SoftwareDriverTarget || + s32([SoftwareDriverTarget size].width) != surface->getDimension().Width || + s32([SoftwareDriverTarget size].height) != surface->getDimension().Height; + + NSRect areaRect = NSMakeRect(0.0, 0.0, surface->getDimension().Width, surface->getDimension().Height); + const u32 destPitch = (colorSamples * areaRect.size.width); + + // create / update the target + if (updateSize) + { + [SoftwareDriverTarget release]; + // allocate target for IImage + SoftwareDriverTarget = [[NSBitmapImageRep alloc] + initWithBitmapDataPlanes: nil + pixelsWide: areaRect.size.width + pixelsHigh: areaRect.size.height + bitsPerSample: 8 + samplesPerPixel: colorSamples + hasAlpha: NO + isPlanar: NO + colorSpaceName: NSCalibratedRGBColorSpace + bytesPerRow: destPitch + bitsPerPixel: 8*colorSamples]; + } + + if (SoftwareDriverTarget==nil) + return false; + + // get pointer to image data + unsigned char* imgData = (unsigned char*)surface->lock(); + + u8* srcdata = reinterpret_cast(imgData); + u8* destData = reinterpret_cast([SoftwareDriverTarget bitmapData]); + const u32 srcheight = core::min_(surface->getDimension().Height, (u32)areaRect.size.height); + const u32 srcPitch = surface->getPitch(); + const u32 minWidth = core::min_(surface->getDimension().Width, (u32)areaRect.size.width); + for (u32 y=0; y!=srcheight; ++y) + { + if(SoftwareRendererType == 2) + { + if (surface->getColorFormat() == video::ECF_A8R8G8B8) + video::CColorConverter::convert_A8R8G8B8toB8G8R8(srcdata, minWidth, destData); + else if (surface->getColorFormat() == video::ECF_A1R5G5B5) + video::CColorConverter::convert_A1R5G5B5toB8G8R8(srcdata, minWidth, destData); + else + video::CColorConverter::convert_viaFormat(srcdata, surface->getColorFormat(), minWidth, destData, video::ECF_R8G8B8); + } + else + { + if (surface->getColorFormat() == video::ECF_A8R8G8B8) + video::CColorConverter::convert_A8R8G8B8toR8G8B8(srcdata, minWidth, destData); + else if (surface->getColorFormat() == video::ECF_A1R5G5B5) + video::CColorConverter::convert_A1R5G5B5toR8G8B8(srcdata, minWidth, destData); + else + video::CColorConverter::convert_viaFormat(srcdata, surface->getColorFormat(), minWidth, destData, video::ECF_R8G8B8); + } + + srcdata += srcPitch; + destData += destPitch; + } + + // unlock the data + surface->unlock(); + + // todo: draw properly into a sub-view + [SoftwareDriverTarget draw]; + } + + return false; +} + + +#if defined (_IRR_COMPILE_WITH_JOYSTICK_EVENTS_) +static void joystickRemovalCallback(void * target, + IOReturn result, void * refcon, void * sender) +{ + JoystickInfo *joy = (JoystickInfo *) refcon; + joy->removed = 1; +} +#endif // _IRR_COMPILE_WITH_JOYSTICK_EVENTS_ + + +bool CIrrDeviceMacOSX::activateJoysticks(core::array & joystickInfo) +{ +#if defined (_IRR_COMPILE_WITH_JOYSTICK_EVENTS_) + ActiveJoysticks.clear(); + joystickInfo.clear(); + + io_object_t hidObject = 0; + io_iterator_t hidIterator = 0; + IOReturn result = kIOReturnSuccess; + mach_port_t masterPort = 0; + CFMutableDictionaryRef hidDictionaryRef = NULL; + + result = IOMasterPort (bootstrap_port, &masterPort); + if (kIOReturnSuccess != result) + { + os::Printer::log("initialiseJoysticks IOMasterPort failed", ELL_ERROR); + return false; + } + + hidDictionaryRef = IOServiceMatching (kIOHIDDeviceKey); + if (!hidDictionaryRef) + { + os::Printer::log("initialiseJoysticks IOServiceMatching failed", ELL_ERROR); + return false; + } + result = IOServiceGetMatchingServices (masterPort, hidDictionaryRef, &hidIterator); + + if (kIOReturnSuccess != result) + { + os::Printer::log("initialiseJoysticks IOServiceGetMatchingServices failed", ELL_ERROR); + return false; + } + + //no joysticks just return + if (!hidIterator) + return false; + + u32 jindex = 0u; + while ((hidObject = IOIteratorNext (hidIterator))) + { + JoystickInfo info; + + // get dictionary for HID properties + CFMutableDictionaryRef hidProperties = 0; + + kern_return_t kern_result = IORegistryEntryCreateCFProperties (hidObject, &hidProperties, kCFAllocatorDefault, kNilOptions); + if ((kern_result == KERN_SUCCESS) && hidProperties) + { + HRESULT plugInResult = S_OK; + SInt32 score = 0; + IOCFPlugInInterface ** ppPlugInInterface = NULL; + result = IOCreatePlugInInterfaceForService (hidObject, kIOHIDDeviceUserClientTypeID, + kIOCFPlugInInterfaceID, &ppPlugInInterface, &score); + if (kIOReturnSuccess == result) + { + plugInResult = (*ppPlugInInterface)->QueryInterface (ppPlugInInterface, + CFUUIDGetUUIDBytes (kIOHIDDeviceInterfaceID), (void **) &(info.interface)); + if (plugInResult != S_OK) + os::Printer::log("initialiseJoysticks query HID class device interface failed", ELL_ERROR); + (*ppPlugInInterface)->Release(ppPlugInInterface); + } + else + continue; + + if (info.interface != NULL) + { + result = (*(info.interface))->open (info.interface, 0); + if (result == kIOReturnSuccess) + { + (*(info.interface))->setRemovalCallback (info.interface, joystickRemovalCallback, &info, &info); + getJoystickDeviceInfo(hidObject, hidProperties, &info); + + // get elements + CFTypeRef refElementTop = CFDictionaryGetValue (hidProperties, CFSTR(kIOHIDElementKey)); + if (refElementTop) + { + CFTypeID type = CFGetTypeID (refElementTop); + if (type == CFArrayGetTypeID()) + { + CFRange range = {0, CFArrayGetCount ((CFArrayRef)refElementTop)}; + info.numActiveJoysticks = ActiveJoysticks.size(); + CFArrayApplyFunction ((CFArrayRef)refElementTop, range, getJoystickComponentArrayHandler, &info); + } + } + } + else + { + CFRelease (hidProperties); + os::Printer::log("initialiseJoysticks Open interface failed", ELL_ERROR); + continue; + } + + CFRelease (hidProperties); + + result = IOObjectRelease (hidObject); + + if ( (info.usagePage != kHIDPage_GenericDesktop) || + ((info.usage != kHIDUsage_GD_Joystick && + info.usage != kHIDUsage_GD_GamePad && + info.usage != kHIDUsage_GD_MultiAxisController)) ) + { + closeJoystickDevice (&info); + continue; + } + + for (u32 i = 0; i < 6; ++i) + info.persistentData.JoystickEvent.Axis[i] = 0; + + ActiveJoysticks.push_back(info); + + SJoystickInfo returnInfo; + returnInfo.Joystick = jindex; + returnInfo.Axes = info.axes; + //returnInfo.Hats = info.hats; + returnInfo.Buttons = info.buttons; + returnInfo.Name = info.joystickName; + returnInfo.PovHat = SJoystickInfo::POV_HAT_UNKNOWN; + ++ jindex; + + //if (info.hatComp.size()) + // returnInfo.PovHat = SJoystickInfo::POV_HAT_PRESENT; + //else + // returnInfo.PovHat = SJoystickInfo::POV_HAT_ABSENT; + + joystickInfo.push_back(returnInfo); + } + + } + else + { + continue; + } + } + result = IOObjectRelease (hidIterator); + + return true; +#endif // _IRR_COMPILE_WITH_JOYSTICK_EVENTS_ + + return false; +} + +void CIrrDeviceMacOSX::pollJoysticks() +{ +#if defined (_IRR_COMPILE_WITH_JOYSTICK_EVENTS_) + if(0 == ActiveJoysticks.size()) + return; + + u32 joystick; + for (joystick = 0; joystick < ActiveJoysticks.size(); ++joystick) + { + if (ActiveJoysticks[joystick].removed) + continue; + + bool found = false; + ActiveJoysticks[joystick].persistentData.JoystickEvent.Joystick = joystick; + + if (ActiveJoysticks[joystick].interface) + { + for (u32 n = 0; n < ActiveJoysticks[joystick].axisComp.size(); n++) + { + IOReturn result = kIOReturnSuccess; + IOHIDEventStruct hidEvent; + hidEvent.value = 0; + result = (*(ActiveJoysticks[joystick].interface))->getElementValue(ActiveJoysticks[joystick].interface, ActiveJoysticks[joystick].axisComp[n].cookie, &hidEvent); + if (kIOReturnSuccess == result) + { + const f32 min = -32768.0f; + const f32 max = 32767.0f; + const f32 deviceScale = max - min; + const f32 readScale = (f32)ActiveJoysticks[joystick].axisComp[n].maxRead - (f32)ActiveJoysticks[joystick].axisComp[n].minRead; + + if (hidEvent.value < ActiveJoysticks[joystick].axisComp[n].minRead) + ActiveJoysticks[joystick].axisComp[n].minRead = hidEvent.value; + if (hidEvent.value > ActiveJoysticks[joystick].axisComp[n].maxRead) + ActiveJoysticks[joystick].axisComp[n].maxRead = hidEvent.value; + + if (readScale != 0.0f) + hidEvent.value = (int)(((f32)((f32)hidEvent.value - (f32)ActiveJoysticks[joystick].axisComp[n].minRead) * deviceScale / readScale) + min); + + if (ActiveJoysticks[joystick].persistentData.JoystickEvent.Axis[n] != (s16)hidEvent.value) + found = true; + ActiveJoysticks[joystick].persistentData.JoystickEvent.Axis[n] = (s16)hidEvent.value; + } + }//axis check + + for (u32 n = 0; n < ActiveJoysticks[joystick].buttonComp.size(); n++) + { + IOReturn result = kIOReturnSuccess; + IOHIDEventStruct hidEvent; + hidEvent.value = 0; + result = (*(ActiveJoysticks[joystick].interface))->getElementValue(ActiveJoysticks[joystick].interface, ActiveJoysticks[joystick].buttonComp[n].cookie, &hidEvent); + if (kIOReturnSuccess == result) + { + u32 ButtonStates = 0; + + if (hidEvent.value && !((ActiveJoysticks[joystick].persistentData.JoystickEvent.ButtonStates & (1 << n)) ? true : false) ) + found = true; + else if (!hidEvent.value && ((ActiveJoysticks[joystick].persistentData.JoystickEvent.ButtonStates & (1 << n)) ? true : false)) + found = true; + + if (hidEvent.value) + ActiveJoysticks[joystick].persistentData.JoystickEvent.ButtonStates |= (1 << n); + else + ActiveJoysticks[joystick].persistentData.JoystickEvent.ButtonStates &= ~(1 << n); + } + }//button check + //still ToDo..will be done soon :) +/* + for (u32 n = 0; n < ActiveJoysticks[joystick].hatComp.size(); n++) + { + IOReturn result = kIOReturnSuccess; + IOHIDEventStruct hidEvent; + hidEvent.value = 0; + result = (*(ActiveJoysticks[joystick].interface))->getElementValue(ActiveJoysticks[joystick].interface, ActiveJoysticks[joystick].hatComp[n].cookie, &hidEvent); + if (kIOReturnSuccess == result) + { + if (ActiveJoysticks[joystick].persistentData.JoystickEvent.POV != hidEvent.value) + found = true; + ActiveJoysticks[joystick].persistentData.JoystickEvent.POV = hidEvent.value; + } + }//hat check +*/ + } + + if (found) + postEventFromUser(ActiveJoysticks[joystick].persistentData); + } +#endif // _IRR_COMPILE_WITH_JOYSTICK_EVENTS_ +} + +video::IVideoModeList* CIrrDeviceMacOSX::getVideoModeList() +{ + if (!VideoModeList.getVideoModeCount()) + { + CGDirectDisplayID display; + display = CGMainDisplayID(); + +#ifdef __MAC_10_6 + CFArrayRef Modes = CGDisplayCopyAllDisplayModes(display, NULL); + + for(int i = 0; i < CFArrayGetCount(Modes); ++i) + { + CGDisplayModeRef CurrentMode = (CGDisplayModeRef)CFArrayGetValueAtIndex(Modes, i); + + u8 Depth = 0; + + CFStringRef pixEnc = CGDisplayModeCopyPixelEncoding(CurrentMode); + + if(CFStringCompare(pixEnc, CFSTR(IO32BitDirectPixels), kCFCompareCaseInsensitive) == kCFCompareEqualTo) + Depth = 32; + else + if(CFStringCompare(pixEnc, CFSTR(IO16BitDirectPixels), kCFCompareCaseInsensitive) == kCFCompareEqualTo) + Depth = 16; + else + if(CFStringCompare(pixEnc, CFSTR(IO8BitIndexedPixels), kCFCompareCaseInsensitive) == kCFCompareEqualTo) + Depth = 8; + + if(Depth) + { + unsigned int Width = CGDisplayModeGetWidth(CurrentMode); + unsigned int Height = CGDisplayModeGetHeight(CurrentMode); + + VideoModeList.addMode(core::dimension2d(Width, Height), Depth); + } + } +#else + CFArrayRef availableModes = CGDisplayAvailableModes(display); + unsigned int numberOfAvailableModes = CFArrayGetCount(availableModes); + for (u32 i= 0; i(width, height), + bitsPerPixel); + } +#endif + } + return &VideoModeList; +} + +} // end namespace + +#endif // _IRR_COMPILE_WITH_OSX_DEVICE_ + diff --git a/libraries/irrlicht-1.8/source/Irrlicht/MacOSX/MacOSX_Prefix.pch b/libraries/irrlicht-1.8/source/Irrlicht/MacOSX/MacOSX_Prefix.pch index b3ea109..dad1979 100644 --- a/libraries/irrlicht-1.8/source/Irrlicht/MacOSX/MacOSX_Prefix.pch +++ b/libraries/irrlicht-1.8/source/Irrlicht/MacOSX/MacOSX_Prefix.pch @@ -1,5 +1,5 @@ -// -// Prefix header for all source files of the 'MacOSX' target in the 'MacOSX' project. -// - -#include +// +// Prefix header for all source files of the 'MacOSX' target in the 'MacOSX' project. +// + +#include diff --git a/libraries/irrlicht-1.8/source/Irrlicht/MacOSX/MainMenu.nib/keyedobjects.nib b/libraries/irrlicht-1.8/source/Irrlicht/MacOSX/MainMenu.nib/keyedobjects.nib index 1a56102..5905dcd 100644 Binary files a/libraries/irrlicht-1.8/source/Irrlicht/MacOSX/MainMenu.nib/keyedobjects.nib and b/libraries/irrlicht-1.8/source/Irrlicht/MacOSX/MainMenu.nib/keyedobjects.nib differ diff --git a/libraries/irrlicht-1.8/source/Irrlicht/MacOSX/OSXClipboard.h b/libraries/irrlicht-1.8/source/Irrlicht/MacOSX/OSXClipboard.h index 68f598d..257cfb4 100644 --- a/libraries/irrlicht-1.8/source/Irrlicht/MacOSX/OSXClipboard.h +++ b/libraries/irrlicht-1.8/source/Irrlicht/MacOSX/OSXClipboard.h @@ -1,15 +1,15 @@ -// Copyright (C) 2005-2006 Etienne Petitjean -// Copyright (C) 2007-2012 Christian Stehno -// This file is part of the "Irrlicht Engine". -// For conditions of distribution and use, see copyright notice in Irrlicht.h - -#ifdef __cplusplus -extern "C" { -#endif - - void OSXCopyToClipboard(const char *text); - char* OSXCopyFromClipboard(); - -#ifdef __cplusplus -} -#endif +// Copyright (C) 2005-2006 Etienne Petitjean +// Copyright (C) 2007-2012 Christian Stehno +// This file is part of the "Irrlicht Engine". +// For conditions of distribution and use, see copyright notice in Irrlicht.h + +#ifdef __cplusplus +extern "C" { +#endif + + void OSXCopyToClipboard(const char *text); + char* OSXCopyFromClipboard(); + +#ifdef __cplusplus +} +#endif diff --git a/libraries/irrlicht-1.8/source/Irrlicht/MacOSX/OSXClipboard.mm b/libraries/irrlicht-1.8/source/Irrlicht/MacOSX/OSXClipboard.mm index d549911..a9123c7 100644 --- a/libraries/irrlicht-1.8/source/Irrlicht/MacOSX/OSXClipboard.mm +++ b/libraries/irrlicht-1.8/source/Irrlicht/MacOSX/OSXClipboard.mm @@ -1,36 +1,36 @@ -// Copyright (C) 2005-2006 Etienne Petitjean -// Copyright (C) 2007-2012 Christian Stehno -// This file is part of the "Irrlicht Engine". -// For conditions of distribution and use, see copyright notice in Irrlicht.h - -#include "OSXClipboard.h" -#import - -void OSXCopyToClipboard(const char *text) -{ - NSString *str; - NSPasteboard *board; - - if ((text != NULL) && (strlen(text) > 0)) - { - str = [NSString stringWithCString:text encoding:NSWindowsCP1252StringEncoding]; - board = [NSPasteboard generalPasteboard]; - [board declareTypes:[NSArray arrayWithObject:NSStringPboardType] owner:NSApp]; - [board setString:str forType:NSStringPboardType]; - } -} - -char* OSXCopyFromClipboard() -{ - NSString* str; - NSPasteboard* board; - char* result; - - result = NULL; - board = [NSPasteboard generalPasteboard]; - str = [board stringForType:NSStringPboardType]; - if (str != nil) - result = (char*)[str cStringUsingEncoding:NSWindowsCP1252StringEncoding]; - return (result); -} - +// Copyright (C) 2005-2006 Etienne Petitjean +// Copyright (C) 2007-2012 Christian Stehno +// This file is part of the "Irrlicht Engine". +// For conditions of distribution and use, see copyright notice in Irrlicht.h + +#include "OSXClipboard.h" +#import + +void OSXCopyToClipboard(const char *text) +{ + NSString *str; + NSPasteboard *board; + + if ((text != NULL) && (strlen(text) > 0)) + { + str = [NSString stringWithCString:text encoding:NSWindowsCP1252StringEncoding]; + board = [NSPasteboard generalPasteboard]; + [board declareTypes:[NSArray arrayWithObject:NSStringPboardType] owner:NSApp]; + [board setString:str forType:NSStringPboardType]; + } +} + +char* OSXCopyFromClipboard() +{ + NSString* str; + NSPasteboard* board; + char* result; + + result = NULL; + board = [NSPasteboard generalPasteboard]; + str = [board stringForType:NSStringPboardType]; + if (str != nil) + result = (char*)[str cStringUsingEncoding:NSWindowsCP1252StringEncoding]; + return (result); +} + -- cgit v1.1