mirror of
https://github.com/dolphin-emu/dolphin.git
synced 2025-01-09 07:39:26 +01:00
Add support for the mouse cursor and mouse clicking events on OS X.
This commit is contained in:
parent
69b1da915f
commit
2f28d938cf
1
.gitignore
vendored
1
.gitignore
vendored
@ -35,3 +35,4 @@ Source/Core/Common/Src/scmrev.h
|
|||||||
*.ipch
|
*.ipch
|
||||||
.sconsign.dblite
|
.sconsign.dblite
|
||||||
Externals/scons-local/*
|
Externals/scons-local/*
|
||||||
|
.DS_Store
|
||||||
|
@ -1129,6 +1129,8 @@ void CFrame::OnConfigPAD(wxCommandEvent& WXUNUSED (event))
|
|||||||
#if defined(HAVE_X11) && HAVE_X11
|
#if defined(HAVE_X11) && HAVE_X11
|
||||||
Window win = X11Utils::XWindowFromHandle(GetHandle());
|
Window win = X11Utils::XWindowFromHandle(GetHandle());
|
||||||
Pad::Initialize((void *)win);
|
Pad::Initialize((void *)win);
|
||||||
|
#elif defined(__APPLE__)
|
||||||
|
Pad::Initialize((void *)this);
|
||||||
#else
|
#else
|
||||||
Pad::Initialize(GetHandle());
|
Pad::Initialize(GetHandle());
|
||||||
#endif
|
#endif
|
||||||
@ -1153,6 +1155,8 @@ void CFrame::OnConfigWiimote(wxCommandEvent& WXUNUSED (event))
|
|||||||
#if defined(HAVE_X11) && HAVE_X11
|
#if defined(HAVE_X11) && HAVE_X11
|
||||||
Window win = X11Utils::XWindowFromHandle(GetHandle());
|
Window win = X11Utils::XWindowFromHandle(GetHandle());
|
||||||
Wiimote::Initialize((void *)win);
|
Wiimote::Initialize((void *)win);
|
||||||
|
#elif defined(__APPLE__)
|
||||||
|
Wiimote::Initialize((void *)this);
|
||||||
#else
|
#else
|
||||||
Wiimote::Initialize(GetHandle());
|
Wiimote::Initialize(GetHandle());
|
||||||
#endif
|
#endif
|
||||||
|
@ -45,7 +45,7 @@ void ControllerInterface::Initialize()
|
|||||||
ciface::Xlib::Init(m_devices, m_hwnd);
|
ciface::Xlib::Init(m_devices, m_hwnd);
|
||||||
#endif
|
#endif
|
||||||
#ifdef CIFACE_USE_OSX
|
#ifdef CIFACE_USE_OSX
|
||||||
ciface::OSX::Init(m_devices);
|
ciface::OSX::Init(m_devices, m_hwnd);
|
||||||
#endif
|
#endif
|
||||||
#ifdef CIFACE_USE_SDL
|
#ifdef CIFACE_USE_SDL
|
||||||
ciface::SDL::Init(m_devices);
|
ciface::SDL::Init(m_devices);
|
||||||
|
@ -7,7 +7,7 @@ namespace ciface
|
|||||||
namespace OSX
|
namespace OSX
|
||||||
{
|
{
|
||||||
|
|
||||||
void Init(std::vector<ControllerInterface::Device*>& devices);
|
void Init(std::vector<ControllerInterface::Device*>& devices, void *window);
|
||||||
void DeInit();
|
void DeInit();
|
||||||
|
|
||||||
void DeviceElementDebugPrint(const void *, void *);
|
void DeviceElementDebugPrint(const void *, void *);
|
||||||
|
@ -1,5 +1,6 @@
|
|||||||
#include <Foundation/Foundation.h>
|
#include <Foundation/Foundation.h>
|
||||||
#include <IOKit/hid/IOHIDLib.h>
|
#include <IOKit/hid/IOHIDLib.h>
|
||||||
|
#include <Cocoa/Cocoa.h>
|
||||||
|
|
||||||
#include "../ControllerInterface.h"
|
#include "../ControllerInterface.h"
|
||||||
#include "OSX.h"
|
#include "OSX.h"
|
||||||
@ -131,6 +132,7 @@ void DeviceDebugPrint(IOHIDDeviceRef device)
|
|||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void *g_window;
|
||||||
|
|
||||||
static void DeviceMatching_callback(void* inContext,
|
static void DeviceMatching_callback(void* inContext,
|
||||||
IOReturn inResult,
|
IOReturn inResult,
|
||||||
@ -150,7 +152,7 @@ static void DeviceMatching_callback(void* inContext,
|
|||||||
if (IOHIDDeviceConformsTo(inIOHIDDeviceRef,
|
if (IOHIDDeviceConformsTo(inIOHIDDeviceRef,
|
||||||
kHIDPage_GenericDesktop, kHIDUsage_GD_Keyboard))
|
kHIDPage_GenericDesktop, kHIDUsage_GD_Keyboard))
|
||||||
devices->push_back(new Keyboard(inIOHIDDeviceRef,
|
devices->push_back(new Keyboard(inIOHIDDeviceRef,
|
||||||
name, kbd_name_counts[name]++));
|
name, kbd_name_counts[name]++, g_window));
|
||||||
#if 0
|
#if 0
|
||||||
else if (IOHIDDeviceConformsTo(inIOHIDDeviceRef,
|
else if (IOHIDDeviceConformsTo(inIOHIDDeviceRef,
|
||||||
kHIDPage_GenericDesktop, kHIDUsage_GD_Mouse))
|
kHIDPage_GenericDesktop, kHIDUsage_GD_Mouse))
|
||||||
@ -162,13 +164,15 @@ static void DeviceMatching_callback(void* inContext,
|
|||||||
name, joy_name_counts[name]++));
|
name, joy_name_counts[name]++));
|
||||||
}
|
}
|
||||||
|
|
||||||
void Init(std::vector<ControllerInterface::Device*>& devices)
|
void Init(std::vector<ControllerInterface::Device*>& devices, void *window)
|
||||||
{
|
{
|
||||||
HIDManager = IOHIDManagerCreate(kCFAllocatorDefault,
|
HIDManager = IOHIDManagerCreate(kCFAllocatorDefault,
|
||||||
kIOHIDOptionsTypeNone);
|
kIOHIDOptionsTypeNone);
|
||||||
if (!HIDManager)
|
if (!HIDManager)
|
||||||
NSLog(@"Failed to create HID Manager reference");
|
NSLog(@"Failed to create HID Manager reference");
|
||||||
|
|
||||||
|
g_window = window;
|
||||||
|
|
||||||
IOHIDManagerSetDeviceMatching(HIDManager, NULL);
|
IOHIDManagerSetDeviceMatching(HIDManager, NULL);
|
||||||
|
|
||||||
// Callbacks for acquisition or loss of a matching device
|
// Callbacks for acquisition or loss of a matching device
|
||||||
|
@ -21,21 +21,51 @@ private:
|
|||||||
const IOHIDDeviceRef m_device;
|
const IOHIDDeviceRef m_device;
|
||||||
std::string m_name;
|
std::string m_name;
|
||||||
};
|
};
|
||||||
|
class Cursor : public Input
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
std::string GetName() const;
|
||||||
|
bool IsDetectable() { return false; }
|
||||||
|
Cursor(u8 index, const float& axis, const bool positive) : m_index(index), m_axis(axis), m_positive(positive) {}
|
||||||
|
ControlState GetState() const;
|
||||||
|
private:
|
||||||
|
const float& m_axis;
|
||||||
|
const u8 m_index;
|
||||||
|
const bool m_positive;
|
||||||
|
};
|
||||||
|
class Button : public Input
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
std::string GetName() const;
|
||||||
|
Button(u8 index, const unsigned char& button) : m_index(index), m_button(button) {}
|
||||||
|
ControlState GetState() const;
|
||||||
|
private:
|
||||||
|
const unsigned char& m_button;
|
||||||
|
const u8 m_index;
|
||||||
|
};
|
||||||
|
|
||||||
public:
|
public:
|
||||||
bool UpdateInput();
|
bool UpdateInput();
|
||||||
bool UpdateOutput();
|
bool UpdateOutput();
|
||||||
|
|
||||||
Keyboard(IOHIDDeviceRef device, std::string name, int index);
|
Keyboard(IOHIDDeviceRef device, std::string name, int index, void *window);
|
||||||
|
|
||||||
std::string GetName() const;
|
std::string GetName() const;
|
||||||
std::string GetSource() const;
|
std::string GetSource() const;
|
||||||
int GetId() const;
|
int GetId() const;
|
||||||
|
|
||||||
private:
|
private:
|
||||||
|
struct
|
||||||
|
{
|
||||||
|
float x, y;
|
||||||
|
} m_cursor;
|
||||||
|
|
||||||
const IOHIDDeviceRef m_device;
|
const IOHIDDeviceRef m_device;
|
||||||
const std::string m_device_name;
|
const std::string m_device_name;
|
||||||
int m_index;
|
int m_index;
|
||||||
|
void *m_window;
|
||||||
|
uint32_t m_windowid;
|
||||||
|
unsigned char m_mousebuttons[3];
|
||||||
};
|
};
|
||||||
|
|
||||||
}
|
}
|
||||||
|
@ -1,5 +1,7 @@
|
|||||||
#include <Foundation/Foundation.h>
|
#include <Foundation/Foundation.h>
|
||||||
#include <IOKit/hid/IOHIDLib.h>
|
#include <IOKit/hid/IOHIDLib.h>
|
||||||
|
#include <Cocoa/Cocoa.h>
|
||||||
|
#include <wx/wx.h> // wxWidgets
|
||||||
|
|
||||||
#include "../ControllerInterface.h"
|
#include "../ControllerInterface.h"
|
||||||
#include "OSXKeyboard.h"
|
#include "OSXKeyboard.h"
|
||||||
@ -9,10 +11,11 @@ namespace ciface
|
|||||||
namespace OSX
|
namespace OSX
|
||||||
{
|
{
|
||||||
|
|
||||||
Keyboard::Keyboard(IOHIDDeviceRef device, std::string name, int index)
|
Keyboard::Keyboard(IOHIDDeviceRef device, std::string name, int index, void *window)
|
||||||
: m_device(device)
|
: m_device(device)
|
||||||
, m_device_name(name)
|
, m_device_name(name)
|
||||||
, m_index(index)
|
, m_index(index)
|
||||||
|
, m_window(window)
|
||||||
{
|
{
|
||||||
// This class should only recieve Keyboard or Keypad devices
|
// This class should only recieve Keyboard or Keypad devices
|
||||||
// Now, filter on just the buttons we can handle sanely
|
// Now, filter on just the buttons we can handle sanely
|
||||||
@ -39,10 +42,48 @@ Keyboard::Keyboard(IOHIDDeviceRef device, std::string name, int index)
|
|||||||
}
|
}
|
||||||
CFRelease(elements);
|
CFRelease(elements);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
m_windowid = [[(NSView *)(((wxWindow *)window)->GetHandle()) window] windowNumber];
|
||||||
|
|
||||||
|
// cursor, with a hax for-loop
|
||||||
|
for (unsigned int i=0; i<4; ++i)
|
||||||
|
AddInput(new Cursor(!!(i&2), (&m_cursor.x)[i/2], !!(i&1)));
|
||||||
|
|
||||||
|
for (u8 i = 0; i < sizeof(m_mousebuttons) / sizeof(m_mousebuttons[0]); ++i)
|
||||||
|
AddInput(new Button(i, m_mousebuttons[i]));
|
||||||
}
|
}
|
||||||
|
|
||||||
bool Keyboard::UpdateInput()
|
bool Keyboard::UpdateInput()
|
||||||
{
|
{
|
||||||
|
CGRect bounds = CGRectZero;
|
||||||
|
uint32_t windowid[1] = { m_windowid };
|
||||||
|
CFArrayRef windowArray = CFArrayCreate(NULL, (const void **) windowid, 1, NULL);
|
||||||
|
CFArrayRef windowDescriptions = CGWindowListCreateDescriptionFromArray(windowArray);
|
||||||
|
CFDictionaryRef windowDescription = (CFDictionaryRef) CFArrayGetValueAtIndex((CFArrayRef) windowDescriptions, 0);
|
||||||
|
|
||||||
|
if (CFDictionaryContainsKey(windowDescription, kCGWindowBounds))
|
||||||
|
{
|
||||||
|
CFDictionaryRef boundsDictionary = (CFDictionaryRef) CFDictionaryGetValue(windowDescription, kCGWindowBounds);
|
||||||
|
|
||||||
|
if (boundsDictionary != NULL)
|
||||||
|
CGRectMakeWithDictionaryRepresentation(boundsDictionary, &bounds);
|
||||||
|
}
|
||||||
|
|
||||||
|
CFRelease(windowArray);
|
||||||
|
|
||||||
|
CGEventRef event = CGEventCreate(nil);
|
||||||
|
CGPoint loc = CGEventGetLocation(event);
|
||||||
|
CFRelease(event);
|
||||||
|
|
||||||
|
loc.x -= bounds.origin.x;
|
||||||
|
loc.y -= bounds.origin.y;
|
||||||
|
m_cursor.x = loc.x / bounds.size.width * 2 - 1.0;
|
||||||
|
m_cursor.y = loc.y / bounds.size.height * 2 - 1.0;
|
||||||
|
|
||||||
|
m_mousebuttons[0] = CGEventSourceButtonState(kCGEventSourceStateHIDSystemState, kCGMouseButtonLeft);
|
||||||
|
m_mousebuttons[1] = CGEventSourceButtonState(kCGEventSourceStateHIDSystemState, kCGMouseButtonRight);
|
||||||
|
m_mousebuttons[2] = CGEventSourceButtonState(kCGEventSourceStateHIDSystemState, kCGMouseButtonCenter);
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -201,10 +242,34 @@ ControlState Keyboard::Key::GetState() const
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
ControlState Keyboard::Cursor::GetState() const
|
||||||
|
{
|
||||||
|
return std::max(0.0f, ControlState(m_axis) / (m_positive ? 1.0f : -1.0f));
|
||||||
|
}
|
||||||
|
|
||||||
|
ControlState Keyboard::Button::GetState() const
|
||||||
|
{
|
||||||
|
return (m_button != 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
std::string Keyboard::Cursor::GetName() const
|
||||||
|
{
|
||||||
|
static char tmpstr[] = "Cursor ..";
|
||||||
|
tmpstr[7] = (char)('X' + m_index);
|
||||||
|
tmpstr[8] = (m_positive ? '+' : '-');
|
||||||
|
return tmpstr;
|
||||||
|
}
|
||||||
|
|
||||||
|
std::string Keyboard::Button::GetName() const
|
||||||
|
{
|
||||||
|
return std::string("Click ") + char('0' + m_index);
|
||||||
|
}
|
||||||
|
|
||||||
std::string Keyboard::Key::GetName() const
|
std::string Keyboard::Key::GetName() const
|
||||||
{
|
{
|
||||||
return m_name;
|
return m_name;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user