mirror of
https://github.com/fergalmoran/ladybird.git
synced 2026-01-06 08:36:15 +00:00
254 lines
12 KiB
Plaintext
254 lines
12 KiB
Plaintext
/*
|
|
* Copyright (c) 2023, Tim Flynn <trflynn89@serenityos.org>
|
|
*
|
|
* SPDX-License-Identifier: BSD-2-Clause
|
|
*/
|
|
|
|
#include <AK/TypeCasts.h>
|
|
#include <AK/Utf8View.h>
|
|
|
|
#import <Carbon/Carbon.h>
|
|
#import <UI/Event.h>
|
|
#import <Utilities/Conversions.h>
|
|
|
|
namespace Ladybird {
|
|
|
|
static Web::UIEvents::KeyModifier ns_modifiers_to_key_modifiers(NSEventModifierFlags modifier_flags, Optional<Web::UIEvents::MouseButton&> button = {})
|
|
{
|
|
unsigned modifiers = Web::UIEvents::KeyModifier::Mod_None;
|
|
|
|
if ((modifier_flags & NSEventModifierFlagShift) != 0) {
|
|
modifiers |= Web::UIEvents::KeyModifier::Mod_Shift;
|
|
}
|
|
if ((modifier_flags & NSEventModifierFlagControl) != 0) {
|
|
if (button == Web::UIEvents::MouseButton::Primary) {
|
|
*button = Web::UIEvents::MouseButton::Secondary;
|
|
} else {
|
|
modifiers |= Web::UIEvents::KeyModifier::Mod_Ctrl;
|
|
}
|
|
}
|
|
if ((modifier_flags & NSEventModifierFlagOption) != 0) {
|
|
modifiers |= Web::UIEvents::KeyModifier::Mod_Alt;
|
|
}
|
|
if ((modifier_flags & NSEventModifierFlagCommand) != 0) {
|
|
modifiers |= Web::UIEvents::KeyModifier::Mod_Super;
|
|
}
|
|
|
|
return static_cast<Web::UIEvents::KeyModifier>(modifiers);
|
|
}
|
|
|
|
Web::MouseEvent ns_event_to_mouse_event(Web::MouseEvent::Type type, NSEvent* event, NSView* view, NSScrollView* scroll_view, Web::UIEvents::MouseButton button)
|
|
{
|
|
auto position = [view convertPoint:event.locationInWindow fromView:nil];
|
|
auto device_position = ns_point_to_gfx_point(position).to_type<Web::DevicePixels>();
|
|
|
|
auto screen_position = [NSEvent mouseLocation];
|
|
auto device_screen_position = ns_point_to_gfx_point(screen_position).to_type<Web::DevicePixels>();
|
|
|
|
auto modifiers = ns_modifiers_to_key_modifiers(event.modifierFlags, button);
|
|
|
|
int wheel_delta_x = 0;
|
|
int wheel_delta_y = 0;
|
|
|
|
if (type == Web::MouseEvent::Type::MouseDown) {
|
|
if (event.clickCount % 2 == 0) {
|
|
type = Web::MouseEvent::Type::DoubleClick;
|
|
}
|
|
} else if (type == Web::MouseEvent::Type::MouseWheel) {
|
|
CGFloat delta_x = -[event scrollingDeltaX];
|
|
CGFloat delta_y = -[event scrollingDeltaY];
|
|
|
|
if (![event hasPreciseScrollingDeltas]) {
|
|
delta_x *= scroll_view.horizontalLineScroll;
|
|
delta_y *= scroll_view.verticalLineScroll;
|
|
}
|
|
|
|
wheel_delta_x = static_cast<int>(delta_x);
|
|
wheel_delta_y = static_cast<int>(delta_y);
|
|
}
|
|
|
|
return { type, device_position, device_screen_position, button, button, modifiers, wheel_delta_x, wheel_delta_y, nullptr };
|
|
}
|
|
|
|
NSEvent* create_context_menu_mouse_event(NSView* view, Gfx::IntPoint position)
|
|
{
|
|
return create_context_menu_mouse_event(view, gfx_point_to_ns_point(position));
|
|
}
|
|
|
|
NSEvent* create_context_menu_mouse_event(NSView* view, NSPoint position)
|
|
{
|
|
return [NSEvent mouseEventWithType:NSEventTypeRightMouseUp
|
|
location:[view convertPoint:position fromView:nil]
|
|
modifierFlags:0
|
|
timestamp:0
|
|
windowNumber:[[view window] windowNumber]
|
|
context:nil
|
|
eventNumber:1
|
|
clickCount:1
|
|
pressure:1.0];
|
|
}
|
|
|
|
static Web::UIEvents::KeyCode ns_key_code_to_key_code(unsigned short key_code, Web::UIEvents::KeyModifier& modifiers)
|
|
{
|
|
auto augment_modifiers_and_return = [&](auto key, auto modifier) {
|
|
modifiers = static_cast<Web::UIEvents::KeyModifier>(static_cast<unsigned>(modifiers) | modifier);
|
|
return key;
|
|
};
|
|
|
|
// clang-format off
|
|
switch (key_code) {
|
|
case kVK_ANSI_0: return Web::UIEvents::KeyCode::Key_0;
|
|
case kVK_ANSI_1: return Web::UIEvents::KeyCode::Key_1;
|
|
case kVK_ANSI_2: return Web::UIEvents::KeyCode::Key_2;
|
|
case kVK_ANSI_3: return Web::UIEvents::KeyCode::Key_3;
|
|
case kVK_ANSI_4: return Web::UIEvents::KeyCode::Key_4;
|
|
case kVK_ANSI_5: return Web::UIEvents::KeyCode::Key_5;
|
|
case kVK_ANSI_6: return Web::UIEvents::KeyCode::Key_6;
|
|
case kVK_ANSI_7: return Web::UIEvents::KeyCode::Key_7;
|
|
case kVK_ANSI_8: return Web::UIEvents::KeyCode::Key_8;
|
|
case kVK_ANSI_9: return Web::UIEvents::KeyCode::Key_9;
|
|
case kVK_ANSI_A: return Web::UIEvents::KeyCode::Key_A;
|
|
case kVK_ANSI_B: return Web::UIEvents::KeyCode::Key_B;
|
|
case kVK_ANSI_C: return Web::UIEvents::KeyCode::Key_C;
|
|
case kVK_ANSI_D: return Web::UIEvents::KeyCode::Key_D;
|
|
case kVK_ANSI_E: return Web::UIEvents::KeyCode::Key_E;
|
|
case kVK_ANSI_F: return Web::UIEvents::KeyCode::Key_F;
|
|
case kVK_ANSI_G: return Web::UIEvents::KeyCode::Key_G;
|
|
case kVK_ANSI_H: return Web::UIEvents::KeyCode::Key_H;
|
|
case kVK_ANSI_I: return Web::UIEvents::KeyCode::Key_I;
|
|
case kVK_ANSI_J: return Web::UIEvents::KeyCode::Key_J;
|
|
case kVK_ANSI_K: return Web::UIEvents::KeyCode::Key_K;
|
|
case kVK_ANSI_L: return Web::UIEvents::KeyCode::Key_L;
|
|
case kVK_ANSI_M: return Web::UIEvents::KeyCode::Key_M;
|
|
case kVK_ANSI_N: return Web::UIEvents::KeyCode::Key_N;
|
|
case kVK_ANSI_O: return Web::UIEvents::KeyCode::Key_O;
|
|
case kVK_ANSI_P: return Web::UIEvents::KeyCode::Key_P;
|
|
case kVK_ANSI_Q: return Web::UIEvents::KeyCode::Key_Q;
|
|
case kVK_ANSI_R: return Web::UIEvents::KeyCode::Key_R;
|
|
case kVK_ANSI_S: return Web::UIEvents::KeyCode::Key_S;
|
|
case kVK_ANSI_T: return Web::UIEvents::KeyCode::Key_T;
|
|
case kVK_ANSI_U: return Web::UIEvents::KeyCode::Key_U;
|
|
case kVK_ANSI_V: return Web::UIEvents::KeyCode::Key_V;
|
|
case kVK_ANSI_W: return Web::UIEvents::KeyCode::Key_W;
|
|
case kVK_ANSI_X: return Web::UIEvents::KeyCode::Key_X;
|
|
case kVK_ANSI_Y: return Web::UIEvents::KeyCode::Key_Y;
|
|
case kVK_ANSI_Z: return Web::UIEvents::KeyCode::Key_Z;
|
|
case kVK_ANSI_Backslash: return Web::UIEvents::KeyCode::Key_Backslash;
|
|
case kVK_ANSI_Comma: return Web::UIEvents::KeyCode::Key_Comma;
|
|
case kVK_ANSI_Equal: return Web::UIEvents::KeyCode::Key_Equal;
|
|
case kVK_ANSI_Grave: return Web::UIEvents::KeyCode::Key_Backtick;
|
|
case kVK_ANSI_Keypad0: return augment_modifiers_and_return(Web::UIEvents::KeyCode::Key_0, Web::UIEvents::KeyModifier::Mod_Keypad);
|
|
case kVK_ANSI_Keypad1: return augment_modifiers_and_return(Web::UIEvents::KeyCode::Key_1, Web::UIEvents::KeyModifier::Mod_Keypad);
|
|
case kVK_ANSI_Keypad2: return augment_modifiers_and_return(Web::UIEvents::KeyCode::Key_2, Web::UIEvents::KeyModifier::Mod_Keypad);
|
|
case kVK_ANSI_Keypad3: return augment_modifiers_and_return(Web::UIEvents::KeyCode::Key_3, Web::UIEvents::KeyModifier::Mod_Keypad);
|
|
case kVK_ANSI_Keypad4: return augment_modifiers_and_return(Web::UIEvents::KeyCode::Key_4, Web::UIEvents::KeyModifier::Mod_Keypad);
|
|
case kVK_ANSI_Keypad5: return augment_modifiers_and_return(Web::UIEvents::KeyCode::Key_5, Web::UIEvents::KeyModifier::Mod_Keypad);
|
|
case kVK_ANSI_Keypad6: return augment_modifiers_and_return(Web::UIEvents::KeyCode::Key_6, Web::UIEvents::KeyModifier::Mod_Keypad);
|
|
case kVK_ANSI_Keypad7: return augment_modifiers_and_return(Web::UIEvents::KeyCode::Key_7, Web::UIEvents::KeyModifier::Mod_Keypad);
|
|
case kVK_ANSI_Keypad8: return augment_modifiers_and_return(Web::UIEvents::KeyCode::Key_8, Web::UIEvents::KeyModifier::Mod_Keypad);
|
|
case kVK_ANSI_Keypad9: return augment_modifiers_and_return(Web::UIEvents::KeyCode::Key_9, Web::UIEvents::KeyModifier::Mod_Keypad);
|
|
case kVK_ANSI_KeypadClear: return augment_modifiers_and_return(Web::UIEvents::KeyCode::Key_Delete, Web::UIEvents::KeyModifier::Mod_Keypad);
|
|
case kVK_ANSI_KeypadDecimal: return augment_modifiers_and_return(Web::UIEvents::KeyCode::Key_Period, Web::UIEvents::KeyModifier::Mod_Keypad);
|
|
case kVK_ANSI_KeypadDivide: return augment_modifiers_and_return(Web::UIEvents::KeyCode::Key_Slash, Web::UIEvents::KeyModifier::Mod_Keypad);
|
|
case kVK_ANSI_KeypadEnter: return augment_modifiers_and_return(Web::UIEvents::KeyCode::Key_Return, Web::UIEvents::KeyModifier::Mod_Keypad);
|
|
case kVK_ANSI_KeypadEquals: return augment_modifiers_and_return(Web::UIEvents::KeyCode::Key_Equal, Web::UIEvents::KeyModifier::Mod_Keypad);
|
|
case kVK_ANSI_KeypadMinus: return augment_modifiers_and_return(Web::UIEvents::KeyCode::Key_Minus, Web::UIEvents::KeyModifier::Mod_Keypad);
|
|
case kVK_ANSI_KeypadMultiply: return augment_modifiers_and_return(Web::UIEvents::KeyCode::Key_Asterisk, Web::UIEvents::KeyModifier::Mod_Keypad);
|
|
case kVK_ANSI_KeypadPlus: return augment_modifiers_and_return(Web::UIEvents::KeyCode::Key_Plus, Web::UIEvents::KeyModifier::Mod_Keypad);
|
|
case kVK_ANSI_LeftBracket: return Web::UIEvents::KeyCode::Key_LeftBracket;
|
|
case kVK_ANSI_Minus: return Web::UIEvents::KeyCode::Key_Minus;
|
|
case kVK_ANSI_Period: return Web::UIEvents::KeyCode::Key_Period;
|
|
case kVK_ANSI_Quote: return Web::UIEvents::KeyCode::Key_Apostrophe;
|
|
case kVK_ANSI_RightBracket: return Web::UIEvents::KeyCode::Key_RightBracket;
|
|
case kVK_ANSI_Semicolon: return Web::UIEvents::KeyCode::Key_Semicolon;
|
|
case kVK_ANSI_Slash: return Web::UIEvents::KeyCode::Key_Slash;
|
|
case kVK_CapsLock: return Web::UIEvents::KeyCode::Key_CapsLock;
|
|
case kVK_Command: return Web::UIEvents::KeyCode::Key_Super;
|
|
case kVK_Control: return Web::UIEvents::KeyCode::Key_Control;
|
|
case kVK_Delete: return Web::UIEvents::KeyCode::Key_Backspace;
|
|
case kVK_DownArrow: return Web::UIEvents::KeyCode::Key_Down;
|
|
case kVK_End: return Web::UIEvents::KeyCode::Key_End;
|
|
case kVK_Escape: return Web::UIEvents::KeyCode::Key_Escape;
|
|
case kVK_F1: return Web::UIEvents::KeyCode::Key_F1;
|
|
case kVK_F2: return Web::UIEvents::KeyCode::Key_F2;
|
|
case kVK_F3: return Web::UIEvents::KeyCode::Key_F3;
|
|
case kVK_F4: return Web::UIEvents::KeyCode::Key_F4;
|
|
case kVK_F5: return Web::UIEvents::KeyCode::Key_F5;
|
|
case kVK_F6: return Web::UIEvents::KeyCode::Key_F6;
|
|
case kVK_F7: return Web::UIEvents::KeyCode::Key_F7;
|
|
case kVK_F8: return Web::UIEvents::KeyCode::Key_F8;
|
|
case kVK_F9: return Web::UIEvents::KeyCode::Key_F9;
|
|
case kVK_F10: return Web::UIEvents::KeyCode::Key_F10;
|
|
case kVK_F11: return Web::UIEvents::KeyCode::Key_F11;
|
|
case kVK_F12: return Web::UIEvents::KeyCode::Key_F12;
|
|
case kVK_ForwardDelete: return Web::UIEvents::KeyCode::Key_Delete;
|
|
case kVK_Home: return Web::UIEvents::KeyCode::Key_Home;
|
|
case kVK_LeftArrow: return Web::UIEvents::KeyCode::Key_Left;
|
|
case kVK_Option: return Web::UIEvents::KeyCode::Key_Alt;
|
|
case kVK_PageDown: return Web::UIEvents::KeyCode::Key_PageDown;
|
|
case kVK_PageUp: return Web::UIEvents::KeyCode::Key_PageUp;
|
|
case kVK_Return: return Web::UIEvents::KeyCode::Key_Return;
|
|
case kVK_RightArrow: return Web::UIEvents::KeyCode::Key_Right;
|
|
case kVK_RightCommand: return Web::UIEvents::KeyCode::Key_Super; // FIXME: We do not distinguish left-vs-right.
|
|
case kVK_RightControl: return Web::UIEvents::KeyCode::Key_Control; // FIXME: We do not distinguish left-vs-right.
|
|
case kVK_RightOption: return Web::UIEvents::KeyCode::Key_Alt; // FIXME: We do not distinguish left-vs-right.
|
|
case kVK_RightShift: return Web::UIEvents::KeyCode::Key_RightShift;
|
|
case kVK_Shift: return Web::UIEvents::KeyCode::Key_Shift;
|
|
case kVK_Space: return Web::UIEvents::KeyCode::Key_Space;
|
|
case kVK_Tab: return Web::UIEvents::KeyCode::Key_Tab;
|
|
case kVK_UpArrow: return Web::UIEvents::KeyCode::Key_Up;
|
|
default: break;
|
|
}
|
|
// clang-format on
|
|
|
|
return Web::UIEvents::KeyCode::Key_Invalid;
|
|
}
|
|
|
|
class KeyData : public Web::ChromeInputData {
|
|
public:
|
|
explicit KeyData(NSEvent* event)
|
|
: m_event(CFBridgingRetain(event))
|
|
{
|
|
}
|
|
|
|
virtual ~KeyData() override
|
|
{
|
|
if (m_event != nullptr) {
|
|
CFBridgingRelease(m_event);
|
|
}
|
|
}
|
|
|
|
NSEvent* take_event()
|
|
{
|
|
VERIFY(m_event != nullptr);
|
|
|
|
CFTypeRef event = exchange(m_event, nullptr);
|
|
return CFBridgingRelease(event);
|
|
}
|
|
|
|
private:
|
|
CFTypeRef m_event { nullptr };
|
|
};
|
|
|
|
Web::KeyEvent ns_event_to_key_event(Web::KeyEvent::Type type, NSEvent* event)
|
|
{
|
|
auto modifiers = ns_modifiers_to_key_modifiers(event.modifierFlags);
|
|
auto key_code = ns_key_code_to_key_code(event.keyCode, modifiers);
|
|
|
|
auto const* utf8 = [event.characters UTF8String];
|
|
Utf8View utf8_view { StringView { utf8, strlen(utf8) } };
|
|
|
|
// FIXME: WebContent should really support multi-code point key events.
|
|
auto code_point = utf8_view.is_empty() ? 0u : *utf8_view.begin();
|
|
|
|
return { type, key_code, modifiers, code_point, make<KeyData>(event) };
|
|
}
|
|
|
|
NSEvent* key_event_to_ns_event(Web::KeyEvent const& event)
|
|
{
|
|
auto& chrome_data = verify_cast<KeyData>(*event.chrome_data);
|
|
return chrome_data.take_event();
|
|
}
|
|
|
|
}
|