2019-02-12 10:30:24 +00:00
|
|
|
|
2015-08-29 18:23:16 +00:00
|
|
|
#if defined(USE_SDL)
|
|
|
|
#include "types.h"
|
|
|
|
#include "cfg/cfg.h"
|
|
|
|
#include "sdl/sdl.h"
|
2021-03-21 19:07:53 +00:00
|
|
|
#include <SDL_syswm.h>
|
2015-08-29 18:23:16 +00:00
|
|
|
#endif
|
2019-02-12 14:56:44 +00:00
|
|
|
#include "hw/maple/maple_devs.h"
|
2019-02-12 10:30:24 +00:00
|
|
|
#include "sdl_gamepad.h"
|
|
|
|
#include "sdl_keyboard.h"
|
2019-10-18 19:57:08 +00:00
|
|
|
#include "wsi/context.h"
|
2020-04-20 16:52:02 +00:00
|
|
|
#include "emulator.h"
|
2020-12-07 21:25:33 +00:00
|
|
|
#include "stdclass.h"
|
2020-12-08 17:35:46 +00:00
|
|
|
#if !defined(_WIN32) && !defined(__APPLE__)
|
|
|
|
#include "linux-dist/icon.h"
|
|
|
|
#endif
|
2015-08-29 18:23:16 +00:00
|
|
|
|
2019-10-05 09:50:14 +00:00
|
|
|
#ifdef USE_VULKAN
|
2021-03-21 19:07:53 +00:00
|
|
|
#include <SDL_vulkan.h>
|
2019-10-05 09:50:14 +00:00
|
|
|
#endif
|
|
|
|
|
2015-10-11 16:06:06 +00:00
|
|
|
static SDL_Window* window = NULL;
|
|
|
|
|
2015-08-29 18:23:16 +00:00
|
|
|
#ifdef TARGET_PANDORA
|
|
|
|
#define WINDOW_WIDTH 800
|
|
|
|
#else
|
|
|
|
#define WINDOW_WIDTH 640
|
|
|
|
#endif
|
|
|
|
#define WINDOW_HEIGHT 480
|
|
|
|
|
2021-01-21 20:00:46 +00:00
|
|
|
static std::shared_ptr<SDLMouse> sdl_mouse_gamepad;
|
2019-02-21 21:39:26 +00:00
|
|
|
static std::shared_ptr<SDLKbGamepadDevice> sdl_kb_gamepad;
|
2019-02-12 10:30:24 +00:00
|
|
|
static SDLKeyboardDevice* sdl_keyboard = NULL;
|
2020-03-22 13:10:07 +00:00
|
|
|
static bool window_fullscreen;
|
|
|
|
static bool window_maximized;
|
|
|
|
static int window_width = WINDOW_WIDTH;
|
|
|
|
static int window_height = WINDOW_HEIGHT;
|
2021-01-19 20:31:48 +00:00
|
|
|
static bool gameRunning;
|
|
|
|
static bool mouseCaptured;
|
2021-01-21 20:00:46 +00:00
|
|
|
static std::map<u32, std::shared_ptr<SDLMouse>> mice;
|
2015-08-29 18:23:16 +00:00
|
|
|
|
2019-02-12 14:56:44 +00:00
|
|
|
static void sdl_open_joystick(int index)
|
|
|
|
{
|
|
|
|
SDL_Joystick *pJoystick = SDL_JoystickOpen(index);
|
2015-08-29 18:23:16 +00:00
|
|
|
|
2019-02-12 14:56:44 +00:00
|
|
|
if (pJoystick == NULL)
|
|
|
|
{
|
2019-07-01 15:41:15 +00:00
|
|
|
INFO_LOG(INPUT, "SDL: Cannot open joystick %d", index + 1);
|
2019-02-12 14:56:44 +00:00
|
|
|
return;
|
|
|
|
}
|
2021-01-21 20:00:46 +00:00
|
|
|
std::shared_ptr<SDLGamepad> gamepad = std::make_shared<SDLGamepad>(index < MAPLE_PORTS ? index : -1, index, pJoystick);
|
|
|
|
SDLGamepad::AddSDLGamepad(gamepad);
|
2019-02-12 14:56:44 +00:00
|
|
|
}
|
2015-08-29 18:23:16 +00:00
|
|
|
|
2019-02-12 14:56:44 +00:00
|
|
|
static void sdl_close_joystick(SDL_JoystickID instance)
|
|
|
|
{
|
2021-01-21 20:00:46 +00:00
|
|
|
std::shared_ptr<SDLGamepad> gamepad = SDLGamepad::GetSDLGamepad(instance);
|
2019-02-21 13:49:27 +00:00
|
|
|
if (gamepad != NULL)
|
2019-02-23 00:06:10 +00:00
|
|
|
gamepad->close();
|
2019-02-12 14:56:44 +00:00
|
|
|
}
|
2019-01-16 19:48:32 +00:00
|
|
|
|
2021-01-19 20:31:48 +00:00
|
|
|
static void captureMouse(bool capture)
|
|
|
|
{
|
|
|
|
if (window == nullptr || !gameRunning)
|
|
|
|
return;
|
|
|
|
if (!capture)
|
|
|
|
{
|
|
|
|
SDL_SetRelativeMouseMode(SDL_FALSE);
|
|
|
|
SDL_SetWindowTitle(window, "Flycast");
|
|
|
|
mouseCaptured = false;
|
|
|
|
}
|
|
|
|
else if (SDL_SetRelativeMouseMode(SDL_TRUE) == 0)
|
|
|
|
{
|
|
|
|
SDL_SetWindowTitle(window, "Flycast - mouse capture");
|
|
|
|
mouseCaptured = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void emuEventCallback(Event event)
|
|
|
|
{
|
|
|
|
switch (event)
|
|
|
|
{
|
|
|
|
case Event::Pause:
|
|
|
|
gameRunning = false;
|
|
|
|
SDL_SetRelativeMouseMode(SDL_FALSE);
|
|
|
|
SDL_SetWindowTitle(window, "Flycast");
|
|
|
|
break;
|
|
|
|
case Event::Resume:
|
|
|
|
gameRunning = true;
|
|
|
|
captureMouse(mouseCaptured);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-01-21 20:00:46 +00:00
|
|
|
static void clearMice()
|
|
|
|
{
|
|
|
|
for (const auto& pair : mice)
|
|
|
|
GamepadDevice::Unregister(pair.second);
|
|
|
|
mice.clear();
|
|
|
|
}
|
|
|
|
|
|
|
|
static void discoverMice()
|
|
|
|
{
|
|
|
|
clearMice();
|
|
|
|
|
|
|
|
auto defaultMouse = std::make_shared<SDLMouse>();
|
|
|
|
mice[0] = defaultMouse;
|
|
|
|
GamepadDevice::Register(defaultMouse);
|
|
|
|
|
|
|
|
#ifdef _WIN32
|
|
|
|
u32 numDevices;
|
|
|
|
GetRawInputDeviceList(NULL, &numDevices, sizeof(RAWINPUTDEVICELIST));
|
|
|
|
if (numDevices > 0)
|
|
|
|
{
|
|
|
|
RAWINPUTDEVICELIST *deviceList;
|
|
|
|
deviceList = new RAWINPUTDEVICELIST[numDevices];
|
|
|
|
if (deviceList != nullptr)
|
|
|
|
{
|
|
|
|
GetRawInputDeviceList(deviceList, &numDevices, sizeof(RAWINPUTDEVICELIST));
|
|
|
|
for (u32 i = 0; i < numDevices; ++i)
|
|
|
|
{
|
|
|
|
RAWINPUTDEVICELIST& device = deviceList[i];
|
|
|
|
if (device.dwType == RIM_TYPEMOUSE)
|
|
|
|
{
|
|
|
|
// Get the device name
|
|
|
|
std::string name;
|
|
|
|
std::string uniqueId;
|
|
|
|
u32 size;
|
|
|
|
GetRawInputDeviceInfo(device.hDevice, RIDI_DEVICENAME, nullptr, &size);
|
|
|
|
if (size > 0)
|
|
|
|
{
|
|
|
|
std::vector<char> deviceNameData(size);
|
|
|
|
u32 res = GetRawInputDeviceInfo(device.hDevice, RIDI_DEVICENAME, &deviceNameData[0], &size);
|
|
|
|
if (res != (u32)-1)
|
|
|
|
{
|
|
|
|
std::string deviceName(&deviceNameData[0], std::strlen(&deviceNameData[0]));
|
|
|
|
name = "Mouse " + deviceName;
|
|
|
|
uniqueId = "sdl_mouse_" + deviceName;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
u32 handle = (u32)(uintptr_t)device.hDevice;
|
|
|
|
if (name.empty())
|
|
|
|
name = "Mouse " + std::to_string(handle);
|
|
|
|
if (uniqueId.empty())
|
|
|
|
uniqueId = "sdl_mouse_" + std::to_string(handle);
|
|
|
|
|
|
|
|
auto ptr = std::make_shared<SDLMouse>(mice.size() >= 4 ? 3 : mice.size(), name, uniqueId, handle);
|
|
|
|
mice[handle] = ptr;
|
|
|
|
GamepadDevice::Register(ptr);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
delete [] deviceList;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
static std::shared_ptr<SDLMouse> getMouse(u32 handle)
|
|
|
|
{
|
|
|
|
auto it = mice.find(handle);
|
|
|
|
if (it != mice.end())
|
|
|
|
return it->second;
|
|
|
|
else
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
2015-08-29 18:23:16 +00:00
|
|
|
void input_sdl_init()
|
|
|
|
{
|
|
|
|
if (SDL_WasInit(SDL_INIT_JOYSTICK) == 0)
|
|
|
|
{
|
2020-04-14 15:44:27 +00:00
|
|
|
// We want joystick events even if we loose focus
|
|
|
|
SDL_SetHint(SDL_HINT_JOYSTICK_ALLOW_BACKGROUND_EVENTS, "1");
|
2020-12-24 07:54:06 +00:00
|
|
|
#ifdef _WIN32
|
|
|
|
if (cfgLoadBool("input", "DisableXInput", false))
|
2020-12-26 08:58:53 +00:00
|
|
|
{
|
|
|
|
// Disable XInput for some old joysticks
|
|
|
|
NOTICE_LOG(INPUT, "Disabling XInput, using DirectInput");
|
2020-12-24 07:54:06 +00:00
|
|
|
SDL_SetHint(SDL_HINT_XINPUT_ENABLED, "0");
|
2020-12-26 08:58:53 +00:00
|
|
|
}
|
2020-12-24 07:54:06 +00:00
|
|
|
#endif
|
2018-09-03 17:23:22 +00:00
|
|
|
if (SDL_InitSubSystem(SDL_INIT_JOYSTICK) < 0)
|
2019-01-16 19:48:32 +00:00
|
|
|
die("SDL: error initializing Joystick subsystem");
|
2020-12-07 21:18:23 +00:00
|
|
|
|
|
|
|
std::string db = get_readonly_data_path("gamecontrollerdb.txt");
|
|
|
|
int rv = SDL_GameControllerAddMappingsFromFile(db.c_str());
|
|
|
|
if (rv < 0)
|
|
|
|
{
|
|
|
|
db = get_readonly_config_path("gamecontrollerdb.txt");
|
|
|
|
rv = SDL_GameControllerAddMappingsFromFile(db.c_str());
|
2015-08-29 18:23:16 +00:00
|
|
|
}
|
2020-12-07 21:18:23 +00:00
|
|
|
if (rv > 0)
|
|
|
|
DEBUG_LOG(INPUT ,"%d mappings loaded from %s", rv, db.c_str());
|
2015-08-29 18:23:16 +00:00
|
|
|
}
|
2019-02-23 00:06:10 +00:00
|
|
|
if (SDL_WasInit(SDL_INIT_HAPTIC) == 0)
|
|
|
|
SDL_InitSubSystem(SDL_INIT_HAPTIC);
|
2019-01-16 19:48:32 +00:00
|
|
|
|
2020-04-26 08:03:57 +00:00
|
|
|
#if !defined(__APPLE__)
|
2019-02-07 22:44:30 +00:00
|
|
|
SDL_SetRelativeMouseMode(SDL_FALSE);
|
2019-01-16 19:48:32 +00:00
|
|
|
|
2019-02-16 12:52:38 +00:00
|
|
|
sdl_keyboard = new SDLKeyboardDevice(0);
|
2019-02-21 21:39:26 +00:00
|
|
|
sdl_kb_gamepad = std::make_shared<SDLKbGamepadDevice>(0);
|
|
|
|
GamepadDevice::Register(sdl_kb_gamepad);
|
2021-01-21 20:00:46 +00:00
|
|
|
discoverMice();
|
2021-01-19 20:31:48 +00:00
|
|
|
|
|
|
|
EventManager::listen(Event::Pause, emuEventCallback);
|
|
|
|
EventManager::listen(Event::Resume, emuEventCallback);
|
2019-08-14 07:20:24 +00:00
|
|
|
#endif
|
2019-02-07 22:44:30 +00:00
|
|
|
}
|
2015-09-12 18:19:28 +00:00
|
|
|
|
2021-01-21 20:00:46 +00:00
|
|
|
inline void SDLMouse::detect_btn_input(input_detected_cb button_pressed)
|
2019-02-07 22:44:30 +00:00
|
|
|
{
|
2021-01-21 20:00:46 +00:00
|
|
|
GamepadDevice::detect_btn_input(button_pressed);
|
|
|
|
if (rawHandle != 0)
|
|
|
|
{
|
|
|
|
auto defaultMouse = getMouse(0);
|
|
|
|
defaultMouse->detectedRawMouse = getMouse(rawHandle);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
inline void SDLMouse::cancel_detect_input()
|
|
|
|
{
|
|
|
|
GamepadDevice::cancel_detect_input();
|
|
|
|
if (rawHandle != 0)
|
|
|
|
{
|
|
|
|
auto defaultMouse = getMouse(0);
|
|
|
|
defaultMouse->detectedRawMouse = nullptr;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
inline void SDLMouse::setMouseAbsPos(int x, int y) {
|
|
|
|
if (maple_port() < 0)
|
|
|
|
return;
|
|
|
|
|
2019-02-07 22:44:30 +00:00
|
|
|
int width, height;
|
|
|
|
SDL_GetWindowSize(window, &width, &height);
|
|
|
|
if (width != 0 && height != 0)
|
2021-01-21 20:00:46 +00:00
|
|
|
SetMousePosition(x, y, width, height, maple_port());
|
|
|
|
}
|
|
|
|
|
|
|
|
inline void SDLMouse::setMouseRelPos(int deltax, int deltay) {
|
|
|
|
if (maple_port() < 0)
|
|
|
|
return;
|
|
|
|
SetRelativeMousePosition(deltax, deltay, maple_port());
|
|
|
|
}
|
|
|
|
|
|
|
|
#define SET_FLAG(field, mask, expr) (field) = ((expr) ? ((field) & ~(mask)) : ((field) | (mask)))
|
|
|
|
|
|
|
|
inline void SDLMouse::setMouseButton(u32 button, bool pressed) {
|
|
|
|
if (maple_port() < 0)
|
|
|
|
return;
|
|
|
|
|
|
|
|
switch (button)
|
|
|
|
{
|
|
|
|
case SDL_BUTTON_LEFT:
|
|
|
|
SET_FLAG(mo_buttons[maple_port()], 1 << 2, pressed);
|
|
|
|
break;
|
|
|
|
case SDL_BUTTON_RIGHT:
|
|
|
|
SET_FLAG(mo_buttons[maple_port()], 1 << 1, pressed);
|
|
|
|
break;
|
|
|
|
case SDL_BUTTON_MIDDLE:
|
|
|
|
SET_FLAG(mo_buttons[maple_port()], 1 << 3, pressed);
|
|
|
|
break;
|
|
|
|
}
|
2015-08-29 18:23:16 +00:00
|
|
|
}
|
|
|
|
|
2020-12-02 13:40:50 +00:00
|
|
|
void input_sdl_handle()
|
2015-08-29 18:23:16 +00:00
|
|
|
{
|
2021-01-21 20:00:46 +00:00
|
|
|
SDLGamepad::UpdateRumble();
|
2019-02-23 00:06:10 +00:00
|
|
|
|
2015-08-29 18:23:16 +00:00
|
|
|
SDL_Event event;
|
|
|
|
while (SDL_PollEvent(&event))
|
|
|
|
{
|
|
|
|
switch (event.type)
|
|
|
|
{
|
2020-04-26 08:03:57 +00:00
|
|
|
#if !defined(__APPLE__)
|
2020-03-22 13:10:07 +00:00
|
|
|
case SDL_QUIT:
|
2019-02-25 16:52:53 +00:00
|
|
|
dc_exit();
|
2018-09-03 17:23:22 +00:00
|
|
|
break;
|
2019-08-14 20:48:34 +00:00
|
|
|
|
2015-08-29 18:23:16 +00:00
|
|
|
case SDL_KEYDOWN:
|
|
|
|
case SDL_KEYUP:
|
2020-05-08 16:41:36 +00:00
|
|
|
if (event.type == SDL_KEYDOWN && event.key.keysym.sym == SDLK_RETURN && (event.key.keysym.mod & KMOD_ALT))
|
2019-02-07 22:44:30 +00:00
|
|
|
{
|
2020-03-22 13:10:07 +00:00
|
|
|
if (window_fullscreen)
|
|
|
|
SDL_SetWindowFullscreen(window, 0);
|
|
|
|
else
|
|
|
|
SDL_SetWindowFullscreen(window, SDL_WINDOW_FULLSCREEN_DESKTOP);
|
|
|
|
window_fullscreen = !window_fullscreen;
|
|
|
|
}
|
2021-01-19 20:31:48 +00:00
|
|
|
else if (event.type == SDL_KEYDOWN && (event.key.keysym.mod & KMOD_LALT) && (event.key.keysym.mod & KMOD_LCTRL))
|
|
|
|
{
|
|
|
|
captureMouse(!mouseCaptured);
|
|
|
|
}
|
2020-03-22 13:10:07 +00:00
|
|
|
else
|
|
|
|
{
|
|
|
|
sdl_kb_gamepad->gamepad_btn_input(event.key.keysym.sym, event.type == SDL_KEYDOWN);
|
2021-03-02 17:45:18 +00:00
|
|
|
sdl_keyboard->keyboard_input(event.key.keysym.scancode, event.type == SDL_KEYDOWN);
|
2015-08-29 18:23:16 +00:00
|
|
|
}
|
|
|
|
break;
|
2019-02-24 12:19:50 +00:00
|
|
|
case SDL_TEXTINPUT:
|
|
|
|
for (int i = 0; event.text.text[i] != '\0'; i++)
|
|
|
|
sdl_keyboard->keyboard_character(event.text.text[i]);
|
|
|
|
break;
|
2021-01-09 17:16:39 +00:00
|
|
|
case SDL_WINDOWEVENT:
|
|
|
|
if (event.window.event == SDL_WINDOWEVENT_SIZE_CHANGED
|
|
|
|
|| event.window.event == SDL_WINDOWEVENT_RESTORED
|
|
|
|
|| event.window.event == SDL_WINDOWEVENT_MINIMIZED
|
|
|
|
|| event.window.event == SDL_WINDOWEVENT_MAXIMIZED)
|
|
|
|
{
|
2021-04-12 20:49:04 +00:00
|
|
|
#ifdef USE_VULKAN
|
2021-01-09 17:16:39 +00:00
|
|
|
theVulkanContext.SetResized();
|
2021-04-12 20:49:04 +00:00
|
|
|
#endif
|
|
|
|
#ifdef _WIN32
|
|
|
|
theDXContext.resize();
|
|
|
|
#endif
|
2021-01-09 17:16:39 +00:00
|
|
|
}
|
|
|
|
break;
|
2019-08-14 07:20:24 +00:00
|
|
|
#endif
|
2015-08-29 18:23:16 +00:00
|
|
|
case SDL_JOYBUTTONDOWN:
|
|
|
|
case SDL_JOYBUTTONUP:
|
|
|
|
{
|
2021-01-21 20:00:46 +00:00
|
|
|
std::shared_ptr<SDLGamepad> device = SDLGamepad::GetSDLGamepad((SDL_JoystickID)event.jbutton.which);
|
2019-02-12 14:56:44 +00:00
|
|
|
if (device != NULL)
|
|
|
|
device->gamepad_btn_input(event.jbutton.button, event.type == SDL_JOYBUTTONDOWN);
|
2015-08-29 18:23:16 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
case SDL_JOYAXISMOTION:
|
|
|
|
{
|
2021-01-21 20:00:46 +00:00
|
|
|
std::shared_ptr<SDLGamepad> device = SDLGamepad::GetSDLGamepad((SDL_JoystickID)event.jaxis.which);
|
2019-02-12 14:56:44 +00:00
|
|
|
if (device != NULL)
|
|
|
|
device->gamepad_axis_input(event.jaxis.axis, event.jaxis.value);
|
2015-08-29 18:23:16 +00:00
|
|
|
}
|
|
|
|
break;
|
2020-03-22 13:10:07 +00:00
|
|
|
case SDL_JOYHATMOTION:
|
|
|
|
{
|
2021-01-21 20:00:46 +00:00
|
|
|
std::shared_ptr<SDLGamepad> device = SDLGamepad::GetSDLGamepad((SDL_JoystickID)event.jhat.which);
|
2020-03-22 13:10:07 +00:00
|
|
|
if (device != NULL)
|
|
|
|
{
|
|
|
|
u32 hatid = (event.jhat.hat + 1) << 8;
|
|
|
|
if (event.jhat.value & SDL_HAT_UP)
|
|
|
|
{
|
|
|
|
device->gamepad_btn_input(hatid + 0, true);
|
|
|
|
device->gamepad_btn_input(hatid + 1, false);
|
|
|
|
}
|
|
|
|
else if (event.jhat.value & SDL_HAT_DOWN)
|
|
|
|
{
|
|
|
|
device->gamepad_btn_input(hatid + 0, false);
|
|
|
|
device->gamepad_btn_input(hatid + 1, true);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
device->gamepad_btn_input(hatid + 0, false);
|
|
|
|
device->gamepad_btn_input(hatid + 1, false);
|
|
|
|
}
|
|
|
|
if (event.jhat.value & SDL_HAT_LEFT)
|
|
|
|
{
|
|
|
|
device->gamepad_btn_input(hatid + 2, true);
|
|
|
|
device->gamepad_btn_input(hatid + 3, false);
|
|
|
|
}
|
|
|
|
else if (event.jhat.value & SDL_HAT_RIGHT)
|
|
|
|
{
|
|
|
|
device->gamepad_btn_input(hatid + 2, false);
|
|
|
|
device->gamepad_btn_input(hatid + 3, true);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
device->gamepad_btn_input(hatid + 2, false);
|
|
|
|
device->gamepad_btn_input(hatid + 3, false);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
2018-09-03 17:23:22 +00:00
|
|
|
|
2020-04-26 08:03:57 +00:00
|
|
|
#if !defined(__APPLE__)
|
2015-08-29 18:23:16 +00:00
|
|
|
case SDL_MOUSEMOTION:
|
2021-01-21 20:00:46 +00:00
|
|
|
{
|
|
|
|
std::shared_ptr<SDLMouse> mouse = getMouse(event.motion.which);
|
|
|
|
if (mouse != nullptr)
|
|
|
|
{
|
|
|
|
if (mouseCaptured && gameRunning)
|
|
|
|
mouse->setMouseRelPos(event.motion.xrel, event.motion.yrel);
|
|
|
|
else
|
|
|
|
mouse->setMouseAbsPos(event.motion.x, event.motion.y);
|
|
|
|
mouse->setMouseButton(SDL_BUTTON_LEFT, event.motion.state & SDL_BUTTON_LMASK);
|
|
|
|
mouse->setMouseButton(SDL_BUTTON_RIGHT, event.motion.state & SDL_BUTTON_RMASK);
|
|
|
|
mouse->setMouseButton(SDL_BUTTON_MIDDLE, event.motion.state & SDL_BUTTON_MMASK);
|
|
|
|
}
|
|
|
|
}
|
2015-08-29 18:23:16 +00:00
|
|
|
break;
|
2019-01-16 19:48:32 +00:00
|
|
|
|
2019-02-07 22:44:30 +00:00
|
|
|
case SDL_MOUSEBUTTONDOWN:
|
|
|
|
case SDL_MOUSEBUTTONUP:
|
|
|
|
{
|
2021-01-21 20:00:46 +00:00
|
|
|
std::shared_ptr<SDLMouse> mouse = getMouse(event.button.which);
|
|
|
|
if (mouse != nullptr)
|
|
|
|
{
|
|
|
|
if (!mouseCaptured || !gameRunning)
|
|
|
|
mouse->setMouseAbsPos(event.button.x, event.button.y);
|
|
|
|
mouse->setMouseButton(event.button.button, event.button.state == SDL_PRESSED);
|
|
|
|
mouse->gamepad_btn_input(event.button.button, event.button.state == SDL_PRESSED);
|
|
|
|
}
|
2019-02-07 22:44:30 +00:00
|
|
|
}
|
2015-08-29 18:23:16 +00:00
|
|
|
break;
|
2019-01-16 19:48:32 +00:00
|
|
|
|
2019-02-24 12:19:50 +00:00
|
|
|
case SDL_MOUSEWHEEL:
|
2021-01-21 20:00:46 +00:00
|
|
|
mo_wheel_delta[0] -= event.wheel.y * 35;
|
2019-02-24 12:19:50 +00:00
|
|
|
break;
|
2019-08-14 07:20:24 +00:00
|
|
|
#endif
|
2019-02-12 14:56:44 +00:00
|
|
|
case SDL_JOYDEVICEADDED:
|
|
|
|
sdl_open_joystick(event.jdevice.which);
|
|
|
|
break;
|
2019-01-16 19:48:32 +00:00
|
|
|
|
2019-02-12 14:56:44 +00:00
|
|
|
case SDL_JOYDEVICEREMOVED:
|
|
|
|
sdl_close_joystick((SDL_JoystickID)event.jdevice.which);
|
|
|
|
break;
|
2015-08-29 18:23:16 +00:00
|
|
|
}
|
|
|
|
}
|
2019-01-16 19:48:32 +00:00
|
|
|
}
|
|
|
|
|
2015-08-29 18:23:16 +00:00
|
|
|
void sdl_window_set_text(const char* text)
|
|
|
|
{
|
2021-01-19 20:31:48 +00:00
|
|
|
if (window != nullptr)
|
|
|
|
SDL_SetWindowTitle(window, text);
|
2015-08-29 18:23:16 +00:00
|
|
|
}
|
|
|
|
|
2020-04-26 08:03:57 +00:00
|
|
|
#if !defined(__APPLE__)
|
2020-03-22 13:10:07 +00:00
|
|
|
static void get_window_state()
|
|
|
|
{
|
|
|
|
u32 flags = SDL_GetWindowFlags(window);
|
|
|
|
window_fullscreen = flags & SDL_WINDOW_FULLSCREEN_DESKTOP;
|
|
|
|
window_maximized = flags & SDL_WINDOW_MAXIMIZED;
|
2021-02-17 22:03:37 +00:00
|
|
|
if (!window_fullscreen && !window_maximized){
|
|
|
|
SDL_GetWindowSize(window, &window_width, &window_height);
|
|
|
|
window_width /= scaling;
|
|
|
|
window_height /= scaling;
|
|
|
|
}
|
|
|
|
|
2020-03-22 13:10:07 +00:00
|
|
|
}
|
|
|
|
|
2021-01-16 18:09:22 +00:00
|
|
|
bool sdl_recreate_window(u32 flags)
|
2015-08-29 18:23:16 +00:00
|
|
|
{
|
2021-02-17 22:03:37 +00:00
|
|
|
#ifdef _WIN32
|
|
|
|
//Enable HiDPI mode in Windows
|
|
|
|
typedef enum PROCESS_DPI_AWARENESS {
|
|
|
|
PROCESS_DPI_UNAWARE = 0,
|
|
|
|
PROCESS_SYSTEM_DPI_AWARE = 1,
|
|
|
|
PROCESS_PER_MONITOR_DPI_AWARE = 2
|
|
|
|
} PROCESS_DPI_AWARENESS;
|
|
|
|
|
|
|
|
HRESULT(WINAPI *SetProcessDpiAwareness)(PROCESS_DPI_AWARENESS dpiAwareness); // Windows 8.1 and later
|
|
|
|
void* shcoreDLL = SDL_LoadObject("SHCORE.DLL");
|
|
|
|
if (shcoreDLL) {
|
|
|
|
SetProcessDpiAwareness = (HRESULT(WINAPI *)(PROCESS_DPI_AWARENESS)) SDL_LoadFunction(shcoreDLL, "SetProcessDpiAwareness");
|
|
|
|
if (SetProcessDpiAwareness) {
|
|
|
|
SetProcessDpiAwareness(PROCESS_PER_MONITOR_DPI_AWARE);
|
|
|
|
|
|
|
|
float ddpi;
|
|
|
|
if (SDL_GetDisplayDPI(0, &ddpi, NULL, NULL) != -1){ //SDL_WINDOWPOS_UNDEFINED is Display 0
|
|
|
|
//When using HiDPI mode, set correct DPI scaling
|
|
|
|
scaling = ddpi/96.f;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2019-10-18 19:57:08 +00:00
|
|
|
int x = SDL_WINDOWPOS_UNDEFINED;
|
|
|
|
int y = SDL_WINDOWPOS_UNDEFINED;
|
2020-03-22 13:10:07 +00:00
|
|
|
window_width = cfgLoadInt("window", "width", window_width);
|
|
|
|
window_height = cfgLoadInt("window", "height", window_height);
|
|
|
|
window_fullscreen = cfgLoadBool("window", "fullscreen", window_fullscreen);
|
|
|
|
window_maximized = cfgLoadBool("window", "maximized", window_maximized);
|
2019-10-18 19:57:08 +00:00
|
|
|
if (window != nullptr)
|
2015-08-29 18:23:16 +00:00
|
|
|
{
|
2019-10-18 19:57:08 +00:00
|
|
|
SDL_GetWindowPosition(window, &x, &y);
|
2020-03-22 13:10:07 +00:00
|
|
|
get_window_state();
|
2019-10-18 19:57:08 +00:00
|
|
|
SDL_DestroyWindow(window);
|
2015-08-29 18:23:16 +00:00
|
|
|
}
|
2021-03-19 14:51:09 +00:00
|
|
|
#if !defined(GLES)
|
|
|
|
flags |= SDL_WINDOW_RESIZABLE;
|
2020-03-22 13:10:07 +00:00
|
|
|
if (window_fullscreen)
|
|
|
|
flags |= SDL_WINDOW_FULLSCREEN_DESKTOP;
|
|
|
|
else if (window_maximized)
|
|
|
|
flags |= SDL_WINDOW_MAXIMIZED;
|
2021-03-19 14:51:09 +00:00
|
|
|
#else
|
|
|
|
flags |= SDL_WINDOW_FULLSCREEN;
|
2019-10-05 09:50:14 +00:00
|
|
|
#endif
|
2021-03-19 14:51:09 +00:00
|
|
|
|
2021-02-17 22:03:37 +00:00
|
|
|
window = SDL_CreateWindow("Flycast", x, y, window_width * scaling, window_height * scaling, flags);
|
2021-01-16 18:09:22 +00:00
|
|
|
if (window == nullptr)
|
|
|
|
{
|
|
|
|
ERROR_LOG(COMMON, "Window creation failed: %s", SDL_GetError());
|
|
|
|
return false;
|
|
|
|
}
|
2021-04-12 20:49:04 +00:00
|
|
|
screen_width = window_width * scaling;
|
|
|
|
screen_height = window_height * scaling;
|
2020-03-22 13:10:07 +00:00
|
|
|
|
2021-03-19 14:51:09 +00:00
|
|
|
#if !defined(GLES) && !defined(_WIN32)
|
2020-12-08 17:35:46 +00:00
|
|
|
// Set the window icon
|
|
|
|
u32 pixels[48 * 48];
|
|
|
|
for (int i = 0; i < 48 * 48; i++)
|
|
|
|
pixels[i] = reicast_icon[i + 2];
|
|
|
|
SDL_Surface *surface = SDL_CreateRGBSurfaceFrom(pixels, 48, 48, 32, 4 * 48, 0x00ff0000, 0x0000ff00, 0x000000ff, 0xff000000);
|
|
|
|
if (surface == NULL)
|
2021-01-16 18:09:22 +00:00
|
|
|
INFO_LOG(COMMON, "Creating icon surface failed: %s", SDL_GetError());
|
2020-12-08 17:35:46 +00:00
|
|
|
else
|
|
|
|
{
|
|
|
|
SDL_SetWindowIcon(window, surface);
|
|
|
|
SDL_FreeSurface(surface);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2019-10-18 19:57:08 +00:00
|
|
|
#ifdef USE_VULKAN
|
|
|
|
theVulkanContext.SetWindow(window, nullptr);
|
2019-10-05 09:50:14 +00:00
|
|
|
#endif
|
2019-10-18 19:57:08 +00:00
|
|
|
theGLContext.SetWindow(window);
|
2021-01-16 18:09:22 +00:00
|
|
|
|
|
|
|
return true;
|
2015-08-29 18:23:16 +00:00
|
|
|
}
|
2015-09-12 18:19:28 +00:00
|
|
|
|
2019-10-18 19:57:08 +00:00
|
|
|
void sdl_window_create()
|
2015-10-11 16:06:06 +00:00
|
|
|
{
|
2019-10-18 19:57:08 +00:00
|
|
|
if (SDL_WasInit(SDL_INIT_VIDEO) == 0)
|
|
|
|
{
|
|
|
|
if (SDL_InitSubSystem(SDL_INIT_VIDEO) != 0)
|
|
|
|
{
|
2020-03-21 14:05:38 +00:00
|
|
|
die("error initializing SDL Video subsystem");
|
2019-10-18 19:57:08 +00:00
|
|
|
}
|
|
|
|
}
|
2019-10-19 16:34:24 +00:00
|
|
|
InitRenderApi();
|
2015-10-11 16:06:06 +00:00
|
|
|
}
|
2019-10-05 09:50:14 +00:00
|
|
|
|
|
|
|
void sdl_window_destroy()
|
|
|
|
{
|
2020-03-22 13:10:07 +00:00
|
|
|
get_window_state();
|
|
|
|
cfgSaveInt("window", "width", window_width);
|
|
|
|
cfgSaveInt("window", "height", window_height);
|
|
|
|
cfgSaveBool("window", "maximized", window_maximized);
|
|
|
|
cfgSaveBool("window", "fullscreen", window_fullscreen);
|
2019-10-18 19:57:08 +00:00
|
|
|
TermRenderApi();
|
2019-10-05 09:50:14 +00:00
|
|
|
SDL_DestroyWindow(window);
|
|
|
|
}
|
2019-12-22 22:49:52 +00:00
|
|
|
|
2020-03-20 15:57:50 +00:00
|
|
|
#ifdef _WIN32
|
2020-03-29 15:32:53 +00:00
|
|
|
#include <windows.h>
|
|
|
|
|
2019-12-22 22:49:52 +00:00
|
|
|
HWND sdl_get_native_hwnd()
|
|
|
|
{
|
|
|
|
SDL_SysWMinfo wmInfo;
|
|
|
|
SDL_VERSION(&wmInfo.version);
|
|
|
|
SDL_GetWindowWMInfo(window, &wmInfo);
|
|
|
|
return wmInfo.info.win.window;
|
|
|
|
}
|
2019-08-14 07:20:24 +00:00
|
|
|
#endif
|
|
|
|
|
2020-04-26 08:03:57 +00:00
|
|
|
#endif // !defined(__APPLE__)
|
2020-03-20 15:57:50 +00:00
|
|
|
|