2024-04-14 17:09:51 +03:00
|
|
|
// SPDX-FileCopyrightText: Copyright 2024 shadPS4 Emulator Project
|
|
|
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
|
|
|
|
2024-06-10 15:24:34 +03:00
|
|
|
#include <SDL3/SDL_events.h>
|
|
|
|
#include <SDL3/SDL_init.h>
|
|
|
|
#include <SDL3/SDL_properties.h>
|
|
|
|
#include <SDL3/SDL_video.h>
|
2024-04-14 17:09:51 +03:00
|
|
|
#include "common/assert.h"
|
2024-06-09 12:25:00 +02:00
|
|
|
#include "common/config.h"
|
2024-04-14 17:09:51 +03:00
|
|
|
#include "common/version.h"
|
|
|
|
#include "core/libraries/pad/pad.h"
|
2024-09-08 16:50:32 -03:00
|
|
|
#include "imgui/renderer/imgui_core.h"
|
2024-04-14 17:09:51 +03:00
|
|
|
#include "input/controller.h"
|
|
|
|
#include "sdl_window.h"
|
2024-07-28 16:54:09 +03:00
|
|
|
#include "video_core/renderdoc.h"
|
2024-04-14 17:09:51 +03:00
|
|
|
|
2024-07-09 02:18:34 -07:00
|
|
|
#ifdef __APPLE__
|
|
|
|
#include <SDL3/SDL_metal.h>
|
|
|
|
#endif
|
|
|
|
|
2024-04-14 17:09:51 +03:00
|
|
|
namespace Frontend {
|
|
|
|
|
2024-07-24 00:12:53 -06:00
|
|
|
WindowSDL::WindowSDL(s32 width_, s32 height_, Input::GameController* controller_,
|
2024-08-01 00:56:10 +03:00
|
|
|
std::string_view window_title)
|
2024-07-25 01:15:44 -06:00
|
|
|
: width{width_}, height{height_}, controller{controller_} {
|
2024-04-14 17:09:51 +03:00
|
|
|
if (SDL_Init(SDL_INIT_VIDEO) < 0) {
|
|
|
|
UNREACHABLE_MSG("Failed to initialize SDL video subsystem: {}", SDL_GetError());
|
|
|
|
}
|
2024-06-14 00:58:57 +03:00
|
|
|
SDL_InitSubSystem(SDL_INIT_AUDIO);
|
2024-04-14 17:09:51 +03:00
|
|
|
|
|
|
|
SDL_PropertiesID props = SDL_CreateProperties();
|
2024-08-01 01:11:58 +03:00
|
|
|
SDL_SetStringProperty(props, SDL_PROP_WINDOW_CREATE_TITLE_STRING,
|
|
|
|
std::string(window_title).c_str());
|
2024-04-14 17:09:51 +03:00
|
|
|
SDL_SetNumberProperty(props, SDL_PROP_WINDOW_CREATE_X_NUMBER, SDL_WINDOWPOS_CENTERED);
|
|
|
|
SDL_SetNumberProperty(props, SDL_PROP_WINDOW_CREATE_Y_NUMBER, SDL_WINDOWPOS_CENTERED);
|
|
|
|
SDL_SetNumberProperty(props, SDL_PROP_WINDOW_CREATE_WIDTH_NUMBER, width);
|
|
|
|
SDL_SetNumberProperty(props, SDL_PROP_WINDOW_CREATE_HEIGHT_NUMBER, height);
|
|
|
|
SDL_SetNumberProperty(props, "flags", SDL_WINDOW_VULKAN);
|
|
|
|
window = SDL_CreateWindowWithProperties(props);
|
|
|
|
SDL_DestroyProperties(props);
|
|
|
|
if (window == nullptr) {
|
|
|
|
UNREACHABLE_MSG("Failed to create window handle: {}", SDL_GetError());
|
|
|
|
}
|
|
|
|
|
2024-06-09 12:25:00 +02:00
|
|
|
SDL_SetWindowFullscreen(window, Config::isFullscreenMode());
|
2024-04-14 17:09:51 +03:00
|
|
|
|
2024-08-13 11:54:08 +02:00
|
|
|
SDL_InitSubSystem(SDL_INIT_GAMEPAD);
|
|
|
|
controller->TryOpenSDLController();
|
|
|
|
|
2024-04-14 17:09:51 +03:00
|
|
|
#if defined(SDL_PLATFORM_WIN32)
|
|
|
|
window_info.type = WindowSystemType::Windows;
|
2024-08-05 19:27:03 +03:00
|
|
|
window_info.render_surface = SDL_GetPointerProperty(SDL_GetWindowProperties(window),
|
|
|
|
SDL_PROP_WINDOW_WIN32_HWND_POINTER, NULL);
|
2024-04-14 17:09:51 +03:00
|
|
|
#elif defined(SDL_PLATFORM_LINUX)
|
|
|
|
if (SDL_strcmp(SDL_GetCurrentVideoDriver(), "x11") == 0) {
|
|
|
|
window_info.type = WindowSystemType::X11;
|
2024-08-05 19:52:30 +03:00
|
|
|
window_info.display_connection = SDL_GetPointerProperty(
|
|
|
|
SDL_GetWindowProperties(window), SDL_PROP_WINDOW_X11_DISPLAY_POINTER, NULL);
|
2024-04-14 17:09:51 +03:00
|
|
|
window_info.render_surface = (void*)SDL_GetNumberProperty(
|
|
|
|
SDL_GetWindowProperties(window), SDL_PROP_WINDOW_X11_WINDOW_NUMBER, 0);
|
|
|
|
} else if (SDL_strcmp(SDL_GetCurrentVideoDriver(), "wayland") == 0) {
|
|
|
|
window_info.type = WindowSystemType::Wayland;
|
2024-08-05 22:36:25 +03:00
|
|
|
window_info.display_connection = SDL_GetPointerProperty(
|
2024-04-14 17:09:51 +03:00
|
|
|
SDL_GetWindowProperties(window), SDL_PROP_WINDOW_WAYLAND_DISPLAY_POINTER, NULL);
|
2024-08-05 22:36:25 +03:00
|
|
|
window_info.render_surface = SDL_GetPointerProperty(
|
|
|
|
SDL_GetWindowProperties(window), SDL_PROP_WINDOW_WAYLAND_SURFACE_POINTER, NULL);
|
2024-04-14 17:09:51 +03:00
|
|
|
}
|
2024-07-09 02:18:34 -07:00
|
|
|
#elif defined(SDL_PLATFORM_MACOS)
|
|
|
|
window_info.type = WindowSystemType::Metal;
|
|
|
|
window_info.render_surface = SDL_Metal_GetLayer(SDL_Metal_CreateView(window));
|
2024-04-14 17:09:51 +03:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
WindowSDL::~WindowSDL() = default;
|
|
|
|
|
|
|
|
void WindowSDL::waitEvent() {
|
|
|
|
// Called on main thread
|
|
|
|
SDL_Event event;
|
|
|
|
|
2024-07-28 16:54:09 +03:00
|
|
|
if (!SDL_WaitEvent(&event)) {
|
2024-04-14 17:09:51 +03:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2024-09-08 16:50:32 -03:00
|
|
|
if (ImGui::Core::ProcessEvent(&event)) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2024-04-14 17:09:51 +03:00
|
|
|
switch (event.type) {
|
|
|
|
case SDL_EVENT_WINDOW_RESIZED:
|
|
|
|
case SDL_EVENT_WINDOW_MAXIMIZED:
|
|
|
|
case SDL_EVENT_WINDOW_RESTORED:
|
|
|
|
onResize();
|
|
|
|
break;
|
|
|
|
case SDL_EVENT_WINDOW_MINIMIZED:
|
|
|
|
case SDL_EVENT_WINDOW_EXPOSED:
|
|
|
|
is_shown = event.type == SDL_EVENT_WINDOW_EXPOSED;
|
|
|
|
onResize();
|
|
|
|
break;
|
|
|
|
case SDL_EVENT_KEY_DOWN:
|
|
|
|
case SDL_EVENT_KEY_UP:
|
|
|
|
onKeyPress(&event);
|
|
|
|
break;
|
2024-08-13 11:54:08 +02:00
|
|
|
case SDL_EVENT_GAMEPAD_BUTTON_DOWN:
|
|
|
|
case SDL_EVENT_GAMEPAD_BUTTON_UP:
|
|
|
|
case SDL_EVENT_GAMEPAD_AXIS_MOTION:
|
2024-08-28 13:48:50 +02:00
|
|
|
case SDL_EVENT_GAMEPAD_ADDED:
|
|
|
|
case SDL_EVENT_GAMEPAD_REMOVED:
|
|
|
|
case SDL_EVENT_GAMEPAD_TOUCHPAD_DOWN:
|
|
|
|
case SDL_EVENT_GAMEPAD_TOUCHPAD_UP:
|
|
|
|
case SDL_EVENT_GAMEPAD_TOUCHPAD_MOTION:
|
2024-08-13 11:54:08 +02:00
|
|
|
onGamepadEvent(&event);
|
|
|
|
break;
|
2024-04-14 17:09:51 +03:00
|
|
|
case SDL_EVENT_QUIT:
|
|
|
|
is_open = false;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-09-11 08:51:18 +03:00
|
|
|
void WindowSDL::setKeysBindingsMap(const std::map<u32, KeysMapping>& bindingsMap) {
|
|
|
|
keysBindingsMap = bindingsMap;
|
|
|
|
}
|
|
|
|
|
2024-04-14 17:09:51 +03:00
|
|
|
void WindowSDL::onResize() {
|
|
|
|
SDL_GetWindowSizeInPixels(window, &width, &height);
|
2024-09-08 16:50:32 -03:00
|
|
|
ImGui::Core::OnResize();
|
2024-04-14 17:09:51 +03:00
|
|
|
}
|
|
|
|
|
2024-09-11 08:51:18 +03:00
|
|
|
using Libraries::Pad::OrbisPadButtonDataOffset;
|
2024-07-17 04:01:28 -07:00
|
|
|
|
2024-09-11 08:51:18 +03:00
|
|
|
void WindowSDL::onKeyPress(const SDL_Event* event) {
|
2024-04-14 17:09:51 +03:00
|
|
|
u32 button = 0;
|
2024-06-17 13:42:39 +03:00
|
|
|
Input::Axis axis = Input::Axis::AxisMax;
|
|
|
|
int axisvalue = 0;
|
|
|
|
int ax = 0;
|
2024-09-11 08:51:18 +03:00
|
|
|
|
|
|
|
bool keyHandlingPending = true;
|
|
|
|
if (!keysBindingsMap.empty()) {
|
|
|
|
|
|
|
|
std::optional<KeysMapping> ps4KeyOpt;
|
|
|
|
auto foundIt = keysBindingsMap.find(event->key.key);
|
|
|
|
if (foundIt != keysBindingsMap.end()) {
|
|
|
|
ps4KeyOpt = foundIt->second;
|
2024-06-21 14:37:49 +03:00
|
|
|
}
|
2024-09-11 08:51:18 +03:00
|
|
|
|
|
|
|
// No support for modifiers (yet)
|
|
|
|
if (ps4KeyOpt.has_value() && (event->key.mod == SDL_KMOD_NONE)) {
|
|
|
|
keyHandlingPending = false;
|
|
|
|
|
|
|
|
auto ps4Key = ps4KeyOpt.value();
|
|
|
|
if (ps4Key == KeysMapping::Start_Key)
|
|
|
|
button = OrbisPadButtonDataOffset::ORBIS_PAD_BUTTON_OPTIONS;
|
|
|
|
if (ps4Key == KeysMapping::Triangle_Key)
|
|
|
|
button = OrbisPadButtonDataOffset::ORBIS_PAD_BUTTON_TRIANGLE;
|
|
|
|
if (ps4Key == KeysMapping::Circle_Key)
|
|
|
|
button = OrbisPadButtonDataOffset::ORBIS_PAD_BUTTON_CIRCLE;
|
|
|
|
if (ps4Key == KeysMapping::Cross_Key)
|
|
|
|
button = OrbisPadButtonDataOffset::ORBIS_PAD_BUTTON_CROSS;
|
|
|
|
if (ps4Key == KeysMapping::Square_Key)
|
|
|
|
button = OrbisPadButtonDataOffset::ORBIS_PAD_BUTTON_SQUARE;
|
|
|
|
if (ps4Key == KeysMapping::R1_Key)
|
|
|
|
button = OrbisPadButtonDataOffset::ORBIS_PAD_BUTTON_R1;
|
|
|
|
if (ps4Key == KeysMapping::R2_Key)
|
|
|
|
handleR2Key(event, button, axis, axisvalue, ax);
|
|
|
|
if (ps4Key == KeysMapping::DPadLeft_Key)
|
|
|
|
button = OrbisPadButtonDataOffset::ORBIS_PAD_BUTTON_LEFT;
|
|
|
|
if (ps4Key == KeysMapping::DPadRight_Key)
|
|
|
|
button = OrbisPadButtonDataOffset::ORBIS_PAD_BUTTON_RIGHT;
|
|
|
|
if (ps4Key == KeysMapping::DPadDown_Key)
|
|
|
|
button = OrbisPadButtonDataOffset::ORBIS_PAD_BUTTON_DOWN;
|
|
|
|
if (ps4Key == KeysMapping::DPadUp_Key)
|
|
|
|
button = OrbisPadButtonDataOffset::ORBIS_PAD_BUTTON_UP;
|
|
|
|
if (ps4Key == KeysMapping::LAnalogLeft_Key)
|
|
|
|
handleLAnalogLeftKey(event, button, axis, axisvalue, ax);
|
|
|
|
if (ps4Key == KeysMapping::LAnalogUp_Key)
|
|
|
|
handleLAnalogUpKey(event, button, axis, axisvalue, ax);
|
|
|
|
if (ps4Key == KeysMapping::LAnalogDown_Key)
|
|
|
|
handleLAnalogDownKey(event, button, axis, axisvalue, ax);
|
|
|
|
if (ps4Key == KeysMapping::RAnalogLeft_Key)
|
|
|
|
handleRAnalogLeftKey(event, button, axis, axisvalue, ax);
|
|
|
|
if (ps4Key == KeysMapping::RAnalogRight_Key)
|
|
|
|
handleRAnalogRightKey(event, button, axis, axisvalue, ax);
|
|
|
|
if (ps4Key == KeysMapping::RAnalogUp_Key)
|
|
|
|
handleRAnalogUpKey(event, button, axis, axisvalue, ax);
|
|
|
|
if (ps4Key == KeysMapping::RAnalogDown_Key)
|
|
|
|
handleRAnalogDownKey(event, button, axis, axisvalue, ax);
|
2024-06-21 14:37:49 +03:00
|
|
|
}
|
2024-09-11 08:51:18 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
if (keyHandlingPending) {
|
|
|
|
switch (event->key.key) {
|
|
|
|
case SDLK_UP:
|
|
|
|
button = OrbisPadButtonDataOffset::ORBIS_PAD_BUTTON_UP;
|
|
|
|
break;
|
|
|
|
case SDLK_DOWN:
|
|
|
|
button = OrbisPadButtonDataOffset::ORBIS_PAD_BUTTON_DOWN;
|
|
|
|
break;
|
|
|
|
case SDLK_LEFT:
|
|
|
|
button = OrbisPadButtonDataOffset::ORBIS_PAD_BUTTON_LEFT;
|
|
|
|
break;
|
|
|
|
case SDLK_RIGHT:
|
|
|
|
button = OrbisPadButtonDataOffset::ORBIS_PAD_BUTTON_RIGHT;
|
|
|
|
break;
|
|
|
|
case Triangle_Key:
|
|
|
|
button = OrbisPadButtonDataOffset::ORBIS_PAD_BUTTON_TRIANGLE;
|
|
|
|
break;
|
|
|
|
case Circle_Key:
|
|
|
|
button = OrbisPadButtonDataOffset::ORBIS_PAD_BUTTON_CIRCLE;
|
|
|
|
break;
|
|
|
|
case Cross_Key:
|
|
|
|
button = OrbisPadButtonDataOffset::ORBIS_PAD_BUTTON_CROSS;
|
|
|
|
break;
|
|
|
|
case Square_Key:
|
|
|
|
button = OrbisPadButtonDataOffset::ORBIS_PAD_BUTTON_SQUARE;
|
|
|
|
break;
|
|
|
|
case SDLK_KP_8:
|
|
|
|
button = OrbisPadButtonDataOffset::ORBIS_PAD_BUTTON_TRIANGLE;
|
|
|
|
break;
|
|
|
|
case SDLK_KP_6:
|
|
|
|
button = OrbisPadButtonDataOffset::ORBIS_PAD_BUTTON_CIRCLE;
|
|
|
|
break;
|
|
|
|
case SDLK_KP_2:
|
|
|
|
button = OrbisPadButtonDataOffset::ORBIS_PAD_BUTTON_CROSS;
|
|
|
|
break;
|
|
|
|
case SDLK_KP_4:
|
|
|
|
button = OrbisPadButtonDataOffset::ORBIS_PAD_BUTTON_SQUARE;
|
|
|
|
break;
|
|
|
|
case SDLK_RETURN:
|
|
|
|
button = OrbisPadButtonDataOffset::ORBIS_PAD_BUTTON_OPTIONS;
|
|
|
|
break;
|
|
|
|
case SDLK_A:
|
|
|
|
handleLAnalogLeftKey(event, button, axis, axisvalue, ax);
|
|
|
|
break;
|
|
|
|
case SDLK_D:
|
|
|
|
handleLAnalogRightKey(event, button, axis, axisvalue, ax);
|
|
|
|
break;
|
|
|
|
case SDLK_W:
|
|
|
|
handleLAnalogUpKey(event, button, axis, axisvalue, ax);
|
|
|
|
break;
|
|
|
|
case SDLK_S:
|
|
|
|
handleLAnalogDownKey(event, button, axis, axisvalue, ax);
|
|
|
|
if (event->key.mod == SDL_KMOD_LCTRL) {
|
|
|
|
// Trigger rdoc capture
|
|
|
|
VideoCore::TriggerCapture();
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case SDLK_J:
|
|
|
|
handleRAnalogLeftKey(event, button, axis, axisvalue, ax);
|
|
|
|
break;
|
|
|
|
case SDLK_L:
|
|
|
|
handleRAnalogRightKey(event, button, axis, axisvalue, ax);
|
|
|
|
break;
|
|
|
|
case SDLK_I:
|
|
|
|
handleRAnalogUpKey(event, button, axis, axisvalue, ax);
|
|
|
|
break;
|
|
|
|
case SDLK_K:
|
|
|
|
handleRAnalogDownKey(event, button, axis, axisvalue, ax);
|
|
|
|
break;
|
|
|
|
case SDLK_X:
|
|
|
|
button = OrbisPadButtonDataOffset::ORBIS_PAD_BUTTON_L3;
|
|
|
|
break;
|
|
|
|
case SDLK_M:
|
|
|
|
button = OrbisPadButtonDataOffset::ORBIS_PAD_BUTTON_R3;
|
|
|
|
break;
|
|
|
|
case SDLK_Q:
|
|
|
|
button = OrbisPadButtonDataOffset::ORBIS_PAD_BUTTON_L1;
|
|
|
|
break;
|
|
|
|
case SDLK_U:
|
|
|
|
button = OrbisPadButtonDataOffset::ORBIS_PAD_BUTTON_R1;
|
|
|
|
break;
|
|
|
|
case SDLK_E:
|
|
|
|
handleL2Key(event, button, axis, axisvalue, ax);
|
|
|
|
break;
|
|
|
|
case SDLK_O:
|
|
|
|
handleR2Key(event, button, axis, axisvalue, ax);
|
|
|
|
break;
|
|
|
|
case SDLK_SPACE:
|
|
|
|
button = OrbisPadButtonDataOffset::ORBIS_PAD_BUTTON_TOUCH_PAD;
|
|
|
|
break;
|
|
|
|
case SDLK_F11:
|
|
|
|
if (event->type == SDL_EVENT_KEY_DOWN) {
|
2024-08-26 18:27:31 -04:00
|
|
|
SDL_WindowFlags flag = SDL_GetWindowFlags(window);
|
|
|
|
bool is_fullscreen = flag & SDL_WINDOW_FULLSCREEN;
|
|
|
|
SDL_SetWindowFullscreen(window, !is_fullscreen);
|
|
|
|
}
|
2024-09-11 08:51:18 +03:00
|
|
|
break;
|
|
|
|
case SDLK_F12:
|
|
|
|
if (event->type == SDL_EVENT_KEY_DOWN) {
|
|
|
|
// Trigger rdoc capture
|
|
|
|
VideoCore::TriggerCapture();
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
2024-08-26 18:27:31 -04:00
|
|
|
}
|
2024-04-14 17:09:51 +03:00
|
|
|
}
|
2024-09-11 08:51:18 +03:00
|
|
|
|
2024-04-14 17:09:51 +03:00
|
|
|
if (button != 0) {
|
2024-06-17 12:52:25 +03:00
|
|
|
controller->CheckButton(0, button, event->type == SDL_EVENT_KEY_DOWN);
|
2024-04-14 17:09:51 +03:00
|
|
|
}
|
2024-06-17 14:33:03 +03:00
|
|
|
if (axis != Input::Axis::AxisMax) {
|
2024-09-11 08:51:18 +03:00
|
|
|
if (event->gaxis.axis == SDL_GAMEPAD_AXIS_LEFT_TRIGGER ||
|
|
|
|
event->gaxis.axis == SDL_GAMEPAD_AXIS_RIGHT_TRIGGER) {
|
|
|
|
controller->Axis(0, axis, Input::GetAxis(0, 0x8000, event->gaxis.value));
|
|
|
|
|
|
|
|
} else {
|
|
|
|
controller->Axis(0, axis, Input::GetAxis(-0x8000, 0x8000, event->gaxis.value));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void WindowSDL::handleR2Key(const SDL_Event* event, u32& button, Input::Axis& axis, int& axisvalue,
|
|
|
|
int& ax) {
|
|
|
|
button = OrbisPadButtonDataOffset::ORBIS_PAD_BUTTON_R2;
|
|
|
|
axis = Input::Axis::TriggerRight;
|
|
|
|
if (event->type == SDL_EVENT_KEY_DOWN) {
|
|
|
|
axisvalue += 255;
|
|
|
|
} else {
|
|
|
|
axisvalue = 0;
|
|
|
|
}
|
|
|
|
ax = Input::GetAxis(0, 0x80, axisvalue);
|
|
|
|
}
|
|
|
|
|
|
|
|
void WindowSDL::handleL2Key(const SDL_Event* event, u32& button, Input::Axis& axis, int& axisvalue,
|
|
|
|
int& ax) {
|
|
|
|
button = OrbisPadButtonDataOffset::ORBIS_PAD_BUTTON_L2;
|
|
|
|
axis = Input::Axis::TriggerLeft;
|
|
|
|
if (event->type == SDL_EVENT_KEY_DOWN) {
|
|
|
|
axisvalue += 255;
|
|
|
|
} else {
|
|
|
|
axisvalue = 0;
|
|
|
|
}
|
|
|
|
ax = Input::GetAxis(0, 0x80, axisvalue);
|
|
|
|
}
|
|
|
|
|
|
|
|
void WindowSDL::handleLAnalogRightKey(const SDL_Event* event, u32& button, Input::Axis& axis,
|
|
|
|
int& axisvalue, int& ax) {
|
|
|
|
axis = Input::Axis::LeftX;
|
|
|
|
if (event->type == SDL_EVENT_KEY_DOWN) {
|
|
|
|
axisvalue += 127;
|
|
|
|
} else {
|
|
|
|
axisvalue = 0;
|
|
|
|
}
|
|
|
|
ax = Input::GetAxis(-0x80, 0x80, axisvalue);
|
|
|
|
}
|
|
|
|
|
|
|
|
void WindowSDL::handleLAnalogLeftKey(const SDL_Event* event, u32& button, Input::Axis& axis,
|
|
|
|
int& axisvalue, int& ax) {
|
|
|
|
axis = Input::Axis::LeftX;
|
|
|
|
if (event->type == SDL_EVENT_KEY_DOWN) {
|
|
|
|
axisvalue += -127;
|
|
|
|
} else {
|
|
|
|
axisvalue = 0;
|
|
|
|
}
|
|
|
|
ax = Input::GetAxis(-0x80, 0x80, axisvalue);
|
|
|
|
}
|
|
|
|
|
|
|
|
void WindowSDL::handleLAnalogUpKey(const SDL_Event* event, u32& button, Input::Axis& axis,
|
|
|
|
int& axisvalue, int& ax) {
|
|
|
|
axis = Input::Axis::LeftY;
|
|
|
|
if (event->type == SDL_EVENT_KEY_DOWN) {
|
|
|
|
axisvalue += -127;
|
|
|
|
} else {
|
|
|
|
axisvalue = 0;
|
|
|
|
}
|
|
|
|
ax = Input::GetAxis(-0x80, 0x80, axisvalue);
|
|
|
|
}
|
|
|
|
|
|
|
|
void WindowSDL::handleLAnalogDownKey(const SDL_Event* event, u32& button, Input::Axis& axis,
|
|
|
|
int& axisvalue, int& ax) {
|
|
|
|
axis = Input::Axis::LeftY;
|
|
|
|
if (event->type == SDL_EVENT_KEY_DOWN) {
|
|
|
|
axisvalue += 127;
|
|
|
|
} else {
|
|
|
|
axisvalue = 0;
|
|
|
|
}
|
|
|
|
ax = Input::GetAxis(-0x80, 0x80, axisvalue);
|
|
|
|
}
|
|
|
|
|
|
|
|
void WindowSDL::handleRAnalogRightKey(const SDL_Event* event, u32& button, Input::Axis& axis,
|
|
|
|
int& axisvalue, int& ax) {
|
|
|
|
axis = Input::Axis::RightX;
|
|
|
|
if (event->type == SDL_EVENT_KEY_DOWN) {
|
|
|
|
axisvalue += 127;
|
|
|
|
} else {
|
|
|
|
axisvalue = 0;
|
|
|
|
}
|
|
|
|
ax = Input::GetAxis(-0x80, 0x80, axisvalue);
|
|
|
|
}
|
|
|
|
|
|
|
|
void WindowSDL::handleRAnalogLeftKey(const SDL_Event* event, u32& button, Input::Axis& axis,
|
|
|
|
int& axisvalue, int& ax) {
|
|
|
|
axis = Input::Axis::RightX;
|
|
|
|
if (event->type == SDL_EVENT_KEY_DOWN) {
|
|
|
|
axisvalue += -127;
|
|
|
|
} else {
|
|
|
|
axisvalue = 0;
|
|
|
|
}
|
|
|
|
ax = Input::GetAxis(-0x80, 0x80, axisvalue);
|
|
|
|
}
|
|
|
|
|
|
|
|
void WindowSDL::handleRAnalogUpKey(const SDL_Event* event, u32& button, Input::Axis& axis,
|
|
|
|
int& axisvalue, int& ax) {
|
|
|
|
axis = Input::Axis::RightY;
|
|
|
|
if (event->type == SDL_EVENT_KEY_DOWN) {
|
|
|
|
axisvalue += -127;
|
|
|
|
} else {
|
|
|
|
axisvalue = 0;
|
|
|
|
}
|
|
|
|
ax = Input::GetAxis(-0x80, 0x80, axisvalue);
|
|
|
|
}
|
|
|
|
|
|
|
|
void WindowSDL::handleRAnalogDownKey(const SDL_Event* event, u32& button, Input::Axis& axis,
|
|
|
|
int& axisvalue, int& ax) {
|
|
|
|
axis = Input::Axis::RightY;
|
|
|
|
if (event->type == SDL_EVENT_KEY_DOWN) {
|
|
|
|
axisvalue += 127;
|
|
|
|
} else {
|
|
|
|
axisvalue = 0;
|
2024-06-17 13:42:39 +03:00
|
|
|
}
|
2024-09-11 08:51:18 +03:00
|
|
|
ax = Input::GetAxis(-0x80, 0x80, axisvalue);
|
2024-04-14 17:09:51 +03:00
|
|
|
}
|
|
|
|
|
2024-08-13 11:54:08 +02:00
|
|
|
void WindowSDL::onGamepadEvent(const SDL_Event* event) {
|
|
|
|
using Libraries::Pad::OrbisPadButtonDataOffset;
|
|
|
|
|
|
|
|
u32 button = 0;
|
|
|
|
Input::Axis axis = Input::Axis::AxisMax;
|
|
|
|
switch (event->type) {
|
2024-08-28 13:48:50 +02:00
|
|
|
case SDL_EVENT_GAMEPAD_ADDED:
|
|
|
|
case SDL_EVENT_GAMEPAD_REMOVED:
|
|
|
|
controller->TryOpenSDLController();
|
|
|
|
break;
|
|
|
|
case SDL_EVENT_GAMEPAD_TOUCHPAD_DOWN:
|
|
|
|
case SDL_EVENT_GAMEPAD_TOUCHPAD_UP:
|
|
|
|
case SDL_EVENT_GAMEPAD_TOUCHPAD_MOTION:
|
|
|
|
controller->SetTouchpadState(event->gtouchpad.finger,
|
|
|
|
event->type != SDL_EVENT_GAMEPAD_TOUCHPAD_UP,
|
|
|
|
event->gtouchpad.x, event->gtouchpad.y);
|
|
|
|
break;
|
2024-08-13 11:54:08 +02:00
|
|
|
case SDL_EVENT_GAMEPAD_BUTTON_DOWN:
|
|
|
|
case SDL_EVENT_GAMEPAD_BUTTON_UP:
|
|
|
|
button = sdlGamepadToOrbisButton(event->gbutton.button);
|
|
|
|
if (button != 0) {
|
|
|
|
controller->CheckButton(0, button, event->type == SDL_EVENT_GAMEPAD_BUTTON_DOWN);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case SDL_EVENT_GAMEPAD_AXIS_MOTION:
|
|
|
|
axis = event->gaxis.axis == SDL_GAMEPAD_AXIS_LEFTX ? Input::Axis::LeftX
|
|
|
|
: event->gaxis.axis == SDL_GAMEPAD_AXIS_LEFTY ? Input::Axis::LeftY
|
|
|
|
: event->gaxis.axis == SDL_GAMEPAD_AXIS_RIGHTX ? Input::Axis::RightX
|
|
|
|
: event->gaxis.axis == SDL_GAMEPAD_AXIS_RIGHTY ? Input::Axis::RightY
|
|
|
|
: event->gaxis.axis == SDL_GAMEPAD_AXIS_LEFT_TRIGGER ? Input::Axis::TriggerLeft
|
|
|
|
: event->gaxis.axis == SDL_GAMEPAD_AXIS_RIGHT_TRIGGER ? Input::Axis::TriggerRight
|
|
|
|
: Input::Axis::AxisMax;
|
|
|
|
if (axis != Input::Axis::AxisMax) {
|
2024-08-22 19:43:45 +03:00
|
|
|
if (event->gaxis.axis == SDL_GAMEPAD_AXIS_LEFT_TRIGGER ||
|
|
|
|
event->gaxis.axis == SDL_GAMEPAD_AXIS_RIGHT_TRIGGER) {
|
|
|
|
controller->Axis(0, axis, Input::GetAxis(0, 0x8000, event->gaxis.value));
|
|
|
|
|
|
|
|
} else {
|
|
|
|
controller->Axis(0, axis, Input::GetAxis(-0x8000, 0x8000, event->gaxis.value));
|
|
|
|
}
|
2024-08-13 11:54:08 +02:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
int WindowSDL::sdlGamepadToOrbisButton(u8 button) {
|
|
|
|
using Libraries::Pad::OrbisPadButtonDataOffset;
|
|
|
|
|
|
|
|
switch (button) {
|
|
|
|
case SDL_GAMEPAD_BUTTON_DPAD_DOWN:
|
|
|
|
return OrbisPadButtonDataOffset::ORBIS_PAD_BUTTON_DOWN;
|
|
|
|
case SDL_GAMEPAD_BUTTON_DPAD_UP:
|
|
|
|
return OrbisPadButtonDataOffset::ORBIS_PAD_BUTTON_UP;
|
|
|
|
case SDL_GAMEPAD_BUTTON_DPAD_LEFT:
|
|
|
|
return OrbisPadButtonDataOffset::ORBIS_PAD_BUTTON_LEFT;
|
|
|
|
case SDL_GAMEPAD_BUTTON_DPAD_RIGHT:
|
|
|
|
return OrbisPadButtonDataOffset::ORBIS_PAD_BUTTON_RIGHT;
|
|
|
|
case SDL_GAMEPAD_BUTTON_SOUTH:
|
|
|
|
return OrbisPadButtonDataOffset::ORBIS_PAD_BUTTON_CROSS;
|
|
|
|
case SDL_GAMEPAD_BUTTON_NORTH:
|
|
|
|
return OrbisPadButtonDataOffset::ORBIS_PAD_BUTTON_TRIANGLE;
|
|
|
|
case SDL_GAMEPAD_BUTTON_WEST:
|
|
|
|
return OrbisPadButtonDataOffset::ORBIS_PAD_BUTTON_SQUARE;
|
|
|
|
case SDL_GAMEPAD_BUTTON_EAST:
|
|
|
|
return OrbisPadButtonDataOffset::ORBIS_PAD_BUTTON_CIRCLE;
|
|
|
|
case SDL_GAMEPAD_BUTTON_START:
|
|
|
|
return OrbisPadButtonDataOffset::ORBIS_PAD_BUTTON_OPTIONS;
|
|
|
|
case SDL_GAMEPAD_BUTTON_TOUCHPAD:
|
|
|
|
return OrbisPadButtonDataOffset::ORBIS_PAD_BUTTON_TOUCH_PAD;
|
|
|
|
case SDL_GAMEPAD_BUTTON_BACK:
|
|
|
|
return OrbisPadButtonDataOffset::ORBIS_PAD_BUTTON_TOUCH_PAD;
|
|
|
|
case SDL_GAMEPAD_BUTTON_LEFT_SHOULDER:
|
|
|
|
return OrbisPadButtonDataOffset::ORBIS_PAD_BUTTON_L1;
|
|
|
|
case SDL_GAMEPAD_BUTTON_RIGHT_SHOULDER:
|
|
|
|
return OrbisPadButtonDataOffset::ORBIS_PAD_BUTTON_R1;
|
|
|
|
case SDL_GAMEPAD_BUTTON_LEFT_STICK:
|
|
|
|
return OrbisPadButtonDataOffset::ORBIS_PAD_BUTTON_L3;
|
|
|
|
case SDL_GAMEPAD_BUTTON_RIGHT_STICK:
|
|
|
|
return OrbisPadButtonDataOffset::ORBIS_PAD_BUTTON_R3;
|
|
|
|
default:
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-09-11 08:51:18 +03:00
|
|
|
} // namespace Frontend
|