ladybird/Libraries/LibWeb/Internals/InternalGamepad.cpp
Luke Wilde 9adf27f009
Some checks failed
CI / macOS, arm64, Sanitizer, Clang (push) Waiting to run
CI / Linux, x86_64, Fuzzers, Clang (push) Waiting to run
CI / Linux, x86_64, Sanitizer, GNU (push) Waiting to run
CI / Linux, x86_64, Sanitizer, Clang (push) Waiting to run
Package the js repl as a binary artifact / Linux, arm64 (push) Waiting to run
Package the js repl as a binary artifact / macOS, arm64 (push) Waiting to run
Package the js repl as a binary artifact / Linux, x86_64 (push) Waiting to run
Run test262 and test-wasm / run_and_update_results (push) Waiting to run
Lint Code / lint (push) Waiting to run
Label PRs with merge conflicts / auto-labeler (push) Waiting to run
Push notes / build (push) Waiting to run
Build Dev Container Image / build (push) Has been cancelled
LibWeb: Add tests for Gamepad API by utilising virtual SDL3 joysticks
2025-09-01 21:10:47 +02:00

178 lines
5.4 KiB
C++

/*
* Copyright (c) 2025, Luke Wilde <luke@ladybird.org>
*
* SPDX-License-Identifier: BSD-2-Clause
*/
#include <LibWeb/Bindings/InternalGamepadPrototype.h>
#include <LibWeb/Bindings/Intrinsics.h>
#include <LibWeb/Internals/InternalGamepad.h>
namespace Web::Internals {
GC_DEFINE_ALLOCATOR(InternalGamepad);
static constexpr Array<SDL_GamepadButton, 15> BUTTONS = {
SDL_GAMEPAD_BUTTON_SOUTH,
SDL_GAMEPAD_BUTTON_EAST,
SDL_GAMEPAD_BUTTON_WEST,
SDL_GAMEPAD_BUTTON_NORTH,
SDL_GAMEPAD_BUTTON_LEFT_SHOULDER,
SDL_GAMEPAD_BUTTON_RIGHT_SHOULDER,
SDL_GAMEPAD_BUTTON_BACK,
SDL_GAMEPAD_BUTTON_START,
SDL_GAMEPAD_BUTTON_LEFT_STICK,
SDL_GAMEPAD_BUTTON_RIGHT_STICK,
SDL_GAMEPAD_BUTTON_DPAD_UP,
SDL_GAMEPAD_BUTTON_DPAD_DOWN,
SDL_GAMEPAD_BUTTON_DPAD_LEFT,
SDL_GAMEPAD_BUTTON_DPAD_RIGHT,
SDL_GAMEPAD_BUTTON_GUIDE,
};
static constexpr Array<SDL_GamepadAxis, 4> AXES {
SDL_GAMEPAD_AXIS_LEFTX,
SDL_GAMEPAD_AXIS_LEFTY,
SDL_GAMEPAD_AXIS_RIGHTX,
SDL_GAMEPAD_AXIS_RIGHTY,
};
static constexpr Array<SDL_GamepadAxis, 2> TRIGGERS {
SDL_GAMEPAD_AXIS_LEFT_TRIGGER,
SDL_GAMEPAD_AXIS_RIGHT_TRIGGER,
};
static constexpr char const* VIRTUAL_GAMEPAD_NAME = "Ladybird Virtual Gamepad";
static SDLCALL bool rumble(void* user_data, u16 low_frequency_rumble, u16 high_frequency_rumble)
{
auto* internal_gamepad = static_cast<InternalGamepad*>(user_data);
internal_gamepad->received_rumble(low_frequency_rumble, high_frequency_rumble);
return true;
}
static SDLCALL bool rumble_triggers(void* user_data, u16 left_rumble, u16 right_rumble)
{
auto* internal_gamepad = static_cast<InternalGamepad*>(user_data);
internal_gamepad->received_rumble_triggers(left_rumble, right_rumble);
return true;
}
InternalGamepad::InternalGamepad(JS::Realm& realm)
: Bindings::PlatformObject(realm)
{
SDL_VirtualJoystickDesc virtual_joystick_desc {};
SDL_INIT_INTERFACE(&virtual_joystick_desc);
virtual_joystick_desc.type = SDL_JOYSTICK_TYPE_GAMEPAD;
virtual_joystick_desc.naxes = AXES.size() + TRIGGERS.size();
virtual_joystick_desc.nbuttons = BUTTONS.size();
u32 button_mask = 0;
for (auto const button : BUTTONS)
button_mask |= 1 << button;
virtual_joystick_desc.button_mask = button_mask;
u32 axis_mask = 0;
for (auto const axis : AXES)
axis_mask |= 1 << axis;
for (auto const trigger : TRIGGERS)
axis_mask |= 1 << trigger;
virtual_joystick_desc.axis_mask = axis_mask;
virtual_joystick_desc.name = VIRTUAL_GAMEPAD_NAME;
virtual_joystick_desc.userdata = this;
virtual_joystick_desc.Rumble = rumble;
virtual_joystick_desc.RumbleTriggers = rumble_triggers;
m_sdl_joystick_id = SDL_AttachVirtualJoystick(&virtual_joystick_desc);
m_sdl_joystick = SDL_OpenJoystick(m_sdl_joystick_id);
}
InternalGamepad::~InternalGamepad() = default;
void InternalGamepad::initialize(JS::Realm& realm)
{
WEB_SET_PROTOTYPE_FOR_INTERFACE(InternalGamepad);
Base::initialize(realm);
}
void InternalGamepad::visit_edges(Cell::Visitor& visitor)
{
Base::visit_edges(visitor);
visitor.visit(m_received_rumble_effects);
visitor.visit(m_received_rumble_trigger_effects);
}
void InternalGamepad::finalize()
{
disconnect();
}
Array<SDL_GamepadButton, 15> const& InternalGamepad::buttons()
{
return BUTTONS;
}
Array<SDL_GamepadAxis, 4> const& InternalGamepad::axes()
{
return AXES;
}
Array<SDL_GamepadAxis, 2> const& InternalGamepad::triggers()
{
return TRIGGERS;
}
void InternalGamepad::set_button(int button, bool down)
{
SDL_SetJoystickVirtualButton(m_sdl_joystick, button, down);
}
void InternalGamepad::set_axis(int axis, short value)
{
SDL_SetJoystickVirtualAxis(m_sdl_joystick, axis, value);
}
GC::RootVector<JS::Object*> InternalGamepad::get_received_rumble_effects() const
{
GC::RootVector<JS::Object*> received_rumble_effects { realm().heap() };
for (auto const received_rumble_effect : m_received_rumble_effects)
received_rumble_effects.append(received_rumble_effect);
return received_rumble_effects;
}
GC::RootVector<JS::Object*> InternalGamepad::get_received_rumble_trigger_effects() const
{
GC::RootVector<JS::Object*> received_rumble_trigger_effects { realm().heap() };
for (auto const received_rumble_trigger_effect : m_received_rumble_trigger_effects)
received_rumble_trigger_effects.append(received_rumble_trigger_effect);
return received_rumble_trigger_effects;
}
void InternalGamepad::received_rumble(u16 low_frequency_rumble, u16 high_frequency_rumble)
{
auto object = JS::Object::create(realm(), nullptr);
object->define_direct_property("lowFrequencyRumble"_utf16, JS::Value(low_frequency_rumble), JS::default_attributes);
object->define_direct_property("highFrequencyRumble"_utf16, JS::Value(high_frequency_rumble), JS::default_attributes);
m_received_rumble_effects.append(object);
}
void InternalGamepad::received_rumble_triggers(u16 left_rumble, u16 right_rumble)
{
auto object = JS::Object::create(realm(), nullptr);
object->define_direct_property("leftRumble"_utf16, JS::Value(left_rumble), JS::default_attributes);
object->define_direct_property("rightRumble"_utf16, JS::Value(right_rumble), JS::default_attributes);
m_received_rumble_trigger_effects.append(object);
}
void InternalGamepad::disconnect()
{
SDL_CloseJoystick(m_sdl_joystick);
SDL_DetachVirtualJoystick(m_sdl_joystick_id);
}
}