| /* |
| * Copyright (C) 2010-2022 Apple Inc. All rights reserved. |
| * |
| * Redistribution and use in source and binary forms, with or without |
| * modification, are permitted provided that the following conditions |
| * are met: |
| * 1. Redistributions of source code must retain the above copyright |
| * notice, this list of conditions and the following disclaimer. |
| * 2. Redistributions in binary form must reproduce the above copyright |
| * notice, this list of conditions and the following disclaimer in the |
| * documentation and/or other materials provided with the distribution. |
| * |
| * THIS SOFTWARE IS PROVIDED BY APPLE INC. AND ITS CONTRIBUTORS ``AS IS'' |
| * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, |
| * THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR |
| * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR ITS CONTRIBUTORS |
| * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR |
| * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF |
| * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS |
| * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN |
| * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) |
| * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF |
| * THE POSSIBILITY OF SUCH DAMAGE. |
| */ |
| |
| #include "config.h" |
| #include "EventSendingController.h" |
| |
| #include "DictionaryFunctions.h" |
| #include "InjectedBundle.h" |
| #include "InjectedBundlePage.h" |
| #include "JSEventSendingController.h" |
| #include <WebKit/WKBundle.h> |
| #include <WebKit/WKBundleFrame.h> |
| #include <WebKit/WKBundlePagePrivate.h> |
| #include <WebKit/WKBundlePrivate.h> |
| #include <WebKit/WKCast.h> |
| #include <WebKit/WKContextMenuItem.h> |
| #include <WebKit/WKMutableDictionary.h> |
| #include <WebKit/WKNumber.h> |
| #include <wtf/StdLibExtras.h> |
| |
| namespace WTR { |
| |
| struct MenuItemPrivateData { |
| MenuItemPrivateData(WKBundlePageRef page, WKContextMenuItemRef item) : |
| m_page(page), |
| m_item(item) { } |
| WKBundlePageRef m_page; |
| WKRetainPtr<WKContextMenuItemRef> m_item; |
| }; |
| |
| #if ENABLE(CONTEXT_MENUS) |
| |
| static JSValueRef menuItemClickCallback(JSContextRef context, JSObjectRef, JSObjectRef thisObject, size_t, const JSValueRef[], JSValueRef*) |
| { |
| MenuItemPrivateData* privateData = static_cast<MenuItemPrivateData*>(JSObjectGetPrivate(thisObject)); |
| WKBundlePageClickMenuItem(privateData->m_page, privateData->m_item.get()); |
| return JSValueMakeUndefined(context); |
| } |
| |
| static JSValueRef getMenuItemTitleCallback(JSContextRef context, JSObjectRef object, JSStringRef propertyName, JSValueRef* exception) |
| { |
| MenuItemPrivateData* privateData = static_cast<MenuItemPrivateData*>(JSObjectGetPrivate(object)); |
| auto wkTitle = adoptWK(WKContextMenuItemCopyTitle(privateData->m_item.get())); |
| return JSValueMakeString(context, toJS(wkTitle).get()); |
| } |
| |
| static JSValueRef getMenuItemEnabledCallback(JSContextRef context, JSObjectRef object, JSStringRef propertyName, JSValueRef* exception) |
| { |
| auto* privateData = static_cast<MenuItemPrivateData*>(JSObjectGetPrivate(object)); |
| return JSValueMakeBoolean(context, WKContextMenuItemGetEnabled(privateData->m_item.get())); |
| } |
| |
| static JSClassRef getMenuItemClass(); |
| |
| static JSValueRef getMenuItemChildrenCallback(JSContextRef context, JSObjectRef object, JSStringRef propertyName, JSValueRef* exception) |
| { |
| MenuItemPrivateData* privateData = static_cast<MenuItemPrivateData*>(JSObjectGetPrivate(object)); |
| auto children = adoptWK(WKContextMenuCopySubmenuItems(privateData->m_item.get())); |
| auto array = JSObjectMakeArray(context, 0, 0, 0); |
| for (size_t i = 0; i < WKArrayGetSize(children.get()); ++i) { |
| auto item = static_cast<WKContextMenuItemRef>(WKArrayGetItemAtIndex(children.get(), i)); |
| auto* privateData = new MenuItemPrivateData(InjectedBundle::singleton().page()->page(), item); |
| JSObjectSetPropertyAtIndex(context, array, i, JSObjectMake(context, getMenuItemClass(), privateData), 0); |
| } |
| return array; |
| } |
| |
| static const JSStaticFunction staticMenuItemFunctions[] = { |
| { "click", menuItemClickCallback, kJSPropertyAttributeReadOnly | kJSPropertyAttributeDontDelete }, |
| { 0, 0, 0 } |
| }; |
| |
| static const JSStaticValue staticMenuItemValues[] = { |
| { "title", getMenuItemTitleCallback, 0, kJSPropertyAttributeReadOnly }, |
| { "children", getMenuItemChildrenCallback, 0, kJSPropertyAttributeReadOnly }, |
| { "enabled", getMenuItemEnabledCallback, 0, kJSPropertyAttributeReadOnly }, |
| { 0, 0, 0, 0 } |
| }; |
| |
| static void staticMenuItemFinalize(JSObjectRef object) |
| { |
| delete static_cast<MenuItemPrivateData*>(JSObjectGetPrivate(object)); |
| } |
| |
| static JSValueRef staticConvertMenuItemToType(JSContextRef context, JSObjectRef object, JSType type, JSValueRef*) |
| { |
| if (kJSTypeString == type) |
| return getMenuItemTitleCallback(context, object, 0, 0); |
| return 0; |
| } |
| |
| static JSClassRef getMenuItemClass() |
| { |
| static JSClassRef menuItemClass = 0; |
| |
| if (!menuItemClass) { |
| JSClassDefinition classDefinition = {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; |
| classDefinition.staticFunctions = staticMenuItemFunctions; |
| classDefinition.staticValues = staticMenuItemValues; |
| classDefinition.finalize = staticMenuItemFinalize; |
| classDefinition.convertToType = staticConvertMenuItemToType; |
| |
| menuItemClass = JSClassCreate(&classDefinition); |
| } |
| |
| return menuItemClass; |
| } |
| |
| #endif |
| |
| static WKEventModifiers parseModifier(const JSRetainPtr<JSStringRef>& modifier) |
| { |
| if (JSStringIsEqualToUTF8CString(modifier.get(), "ctrlKey")) |
| return kWKEventModifiersControlKey; |
| if (JSStringIsEqualToUTF8CString(modifier.get(), "shiftKey") || JSStringIsEqualToUTF8CString(modifier.get(), "rangeSelectionKey")) |
| return kWKEventModifiersShiftKey; |
| if (JSStringIsEqualToUTF8CString(modifier.get(), "altKey")) |
| return kWKEventModifiersAltKey; |
| if (JSStringIsEqualToUTF8CString(modifier.get(), "metaKey")) |
| return kWKEventModifiersMetaKey; |
| if (JSStringIsEqualToUTF8CString(modifier.get(), "capsLockKey")) |
| return kWKEventModifiersCapsLockKey; |
| if (JSStringIsEqualToUTF8CString(modifier.get(), "addSelectionKey")) { |
| #if OS(MACOS) |
| return kWKEventModifiersMetaKey; |
| #else |
| return kWKEventModifiersControlKey; |
| #endif |
| } |
| return 0; |
| } |
| |
| #if ENABLE(TOUCH_EVENTS) |
| |
| static uint64_t parseTouchModifier(JSStringRef modifier) |
| { |
| if (JSStringIsEqualToUTF8CString(modifier, "ctrl")) |
| return kWKEventModifiersControlKey; |
| if (JSStringIsEqualToUTF8CString(modifier, "shift")) |
| return kWKEventModifiersShiftKey; |
| if (JSStringIsEqualToUTF8CString(modifier, "alt")) |
| return kWKEventModifiersAltKey; |
| if (JSStringIsEqualToUTF8CString(modifier, "metaKey")) |
| return kWKEventModifiersMetaKey; |
| return 0; |
| } |
| |
| #endif |
| |
| static WKEventModifiers parseModifierArray(JSContextRef context, JSValueRef arrayValue) |
| { |
| if (!arrayValue) |
| return 0; |
| |
| // The value may either be a string with a single modifier or an array of modifiers. |
| if (JSValueIsString(context, arrayValue)) |
| return parseModifier(createJSString(context, arrayValue)); |
| |
| if (!JSValueIsObject(context, arrayValue)) |
| return 0; |
| JSObjectRef array = const_cast<JSObjectRef>(arrayValue); |
| unsigned length = arrayLength(context, array); |
| WKEventModifiers modifiers = 0; |
| for (unsigned i = 0; i < length; i++) { |
| if (auto value = JSObjectGetPropertyAtIndex(context, array, i, nullptr)) |
| modifiers |= parseModifier(createJSString(context, value)); |
| } |
| return modifiers; |
| } |
| |
| Ref<EventSendingController> EventSendingController::create(uint64_t testIdentifier) |
| { |
| return adoptRef(*new EventSendingController(testIdentifier)); |
| } |
| |
| JSClassRef EventSendingController::wrapperClass() |
| { |
| return JSEventSendingController::eventSendingControllerClass(); |
| } |
| |
| WKRetainPtr<WKMutableDictionaryRef> EventSendingController::createEventSenderDictionary(const char* submessage) |
| { |
| auto body = adoptWK(WKMutableDictionaryCreate()); |
| setValue(body, "TestIdentifier", m_testIdentifier); |
| setValue(body, "SubMessage", submessage); |
| return body; |
| } |
| |
| WKRetainPtr<WKDictionaryRef> EventSendingController::createMouseMessageBody(MouseState state, int button, WKEventModifiers modifiers, JSStringRef pointerType) |
| { |
| auto body = createEventSenderDictionary(state == MouseUp ? "MouseUp" : "MouseDown"); |
| setValue(body, "Button", adoptWK(WKUInt64Create(button))); |
| setValue(body, "Modifiers", adoptWK(WKUInt64Create(modifiers))); |
| if (pointerType) |
| setValue(body, "PointerType", pointerType); |
| return body; |
| } |
| |
| void EventSendingController::mouseDown(JSContextRef context, int button, JSValueRef modifierArray, JSStringRef pointerType) |
| { |
| if (m_isDisabled) |
| return; |
| |
| postSynchronousPageMessage("EventSender", createMouseMessageBody(MouseDown, button, parseModifierArray(context, modifierArray), pointerType)); |
| } |
| |
| void EventSendingController::mouseUp(JSContextRef context, int button, JSValueRef modifierArray, JSStringRef pointerType) |
| { |
| if (m_isDisabled) |
| return; |
| |
| postSynchronousPageMessage("EventSender", createMouseMessageBody(MouseUp, button, parseModifierArray(context, modifierArray), pointerType)); |
| } |
| |
| void EventSendingController::mouseMoveTo(double x, double y, JSStringRef pointerType) |
| { |
| if (m_isDisabled) |
| return; |
| |
| auto body = createEventSenderDictionary("MouseMoveTo"); |
| setValue(body, "X", adoptWK(WKDoubleCreate(x))); |
| setValue(body, "Y", adoptWK(WKDoubleCreate(y))); |
| if (pointerType) |
| setValue(body, "PointerType", pointerType); |
| m_position = WKPointMake(x, y); |
| postSynchronousPageMessage("EventSender", body); |
| |
| WKBundlePageFlushDeferredDidReceiveMouseEventForTesting(InjectedBundle::singleton().pageRef()); |
| auto waitForDidReceiveEventBody = createEventSenderDictionary("WaitForDeferredMouseEvents"); |
| postSynchronousPageMessage("EventSender", waitForDidReceiveEventBody); |
| } |
| |
| void EventSendingController::mouseForceClick() |
| { |
| if (m_isDisabled) |
| return; |
| |
| auto body = createEventSenderDictionary("MouseForceClick"); |
| postSynchronousPageMessage("EventSender", body); |
| } |
| |
| void EventSendingController::startAndCancelMouseForceClick() |
| { |
| if (m_isDisabled) |
| return; |
| |
| auto body = createEventSenderDictionary("StartAndCancelMouseForceClick"); |
| postSynchronousPageMessage("EventSender", body); |
| } |
| |
| void EventSendingController::mouseForceDown() |
| { |
| if (m_isDisabled) |
| return; |
| |
| auto body = createEventSenderDictionary("MouseForceDown"); |
| postSynchronousPageMessage("EventSender", body); |
| } |
| |
| void EventSendingController::mouseForceUp() |
| { |
| if (m_isDisabled) |
| return; |
| |
| auto body = createEventSenderDictionary("MouseForceUp"); |
| postSynchronousPageMessage("EventSender", body); |
| } |
| |
| void EventSendingController::mouseForceChanged(double force) |
| { |
| if (m_isDisabled) |
| return; |
| |
| auto body = createEventSenderDictionary("MouseForceChanged"); |
| setValue(body, "Force", force); |
| postSynchronousPageMessage("EventSender", body); |
| } |
| |
| void EventSendingController::leapForward(int milliseconds) |
| { |
| if (m_isDisabled) |
| return; |
| |
| auto body = createEventSenderDictionary("LeapForward"); |
| setValue(body, "TimeInMilliseconds", adoptWK(WKUInt64Create(milliseconds))); |
| postSynchronousPageMessage("EventSender", body); |
| } |
| |
| void EventSendingController::scheduleAsynchronousClick() |
| { |
| if (m_isDisabled) |
| return; |
| |
| postPageMessage("EventSender", createMouseMessageBody(MouseDown, 0, 0, nullptr)); |
| postPageMessage("EventSender", createMouseMessageBody(MouseUp, 0, 0, nullptr)); |
| } |
| |
| WKRetainPtr<WKMutableDictionaryRef> EventSendingController::createKeyDownMessageBody(JSStringRef key, WKEventModifiers modifiers, int location) |
| { |
| auto body = createEventSenderDictionary("KeyDown"); |
| setValue(body, "Key", key); |
| setValue(body, "Modifiers", adoptWK(WKUInt64Create(modifiers))); |
| setValue(body, "Location", adoptWK(WKUInt64Create(location))); |
| return body; |
| } |
| |
| WKRetainPtr<WKMutableDictionaryRef> EventSendingController::createRawKeyDownMessageBody(JSStringRef key, WKEventModifiers modifiers, int location) |
| { |
| auto body = createEventSenderDictionary("RawKeyDown"); |
| setValue(body, "Key", key); |
| setValue(body, "Modifiers", adoptWK(WKUInt64Create(modifiers))); |
| setValue(body, "Location", adoptWK(WKUInt64Create(location))); |
| return body; |
| } |
| |
| WKRetainPtr<WKMutableDictionaryRef> EventSendingController::createRawKeyUpMessageBody(JSStringRef key, WKEventModifiers modifiers, int location) |
| { |
| auto body = createEventSenderDictionary("RawKeyUp"); |
| setValue(body, "Key", key); |
| setValue(body, "Modifiers", adoptWK(WKUInt64Create(modifiers))); |
| setValue(body, "Location", adoptWK(WKUInt64Create(location))); |
| return body; |
| } |
| |
| void EventSendingController::keyDown(JSContextRef context, JSStringRef key, JSValueRef modifierArray, int location) |
| { |
| if (m_isDisabled) |
| return; |
| |
| postSynchronousPageMessage("EventSender", createKeyDownMessageBody(key, parseModifierArray(context, modifierArray), location)); |
| } |
| |
| void EventSendingController::rawKeyDown(JSContextRef context, JSStringRef key, JSValueRef modifierArray, int location) |
| { |
| if (m_isDisabled) |
| return; |
| |
| postSynchronousPageMessage("EventSender", createRawKeyDownMessageBody(key, parseModifierArray(context, modifierArray), location)); |
| } |
| |
| void EventSendingController::rawKeyUp(JSContextRef context, JSStringRef key, JSValueRef modifierArray, int location) |
| { |
| if (m_isDisabled) |
| return; |
| |
| postSynchronousPageMessage("EventSender", createRawKeyUpMessageBody(key, parseModifierArray(context, modifierArray), location)); |
| } |
| |
| void EventSendingController::scheduleAsynchronousKeyDown(JSStringRef key) |
| { |
| if (m_isDisabled) |
| return; |
| |
| postPageMessage("EventSender", createKeyDownMessageBody(key, 0, 0)); |
| } |
| |
| void EventSendingController::mouseScrollBy(int x, int y) |
| { |
| if (m_isDisabled) |
| return; |
| |
| WKBundlePageForceRepaint(InjectedBundle::singleton().page()->page()); // Triggers a scrolling tree commit. |
| |
| auto body = createEventSenderDictionary("MouseScrollBy"); |
| setValue(body, "X", adoptWK(WKDoubleCreate(x))); |
| setValue(body, "Y", adoptWK(WKDoubleCreate(y))); |
| postPageMessage("EventSender", body); |
| } |
| |
| static uint64_t cgEventPhaseFromString(JSStringRef phaseStr) |
| { |
| if (JSStringIsEqualToUTF8CString(phaseStr, "none")) |
| return 0; |
| if (JSStringIsEqualToUTF8CString(phaseStr, "began")) |
| return 1; // kCGScrollPhaseBegan |
| if (JSStringIsEqualToUTF8CString(phaseStr, "changed")) |
| return 2; // kCGScrollPhaseChanged |
| if (JSStringIsEqualToUTF8CString(phaseStr, "ended")) |
| return 4; // kCGScrollPhaseEnded |
| if (JSStringIsEqualToUTF8CString(phaseStr, "cancelled")) |
| return 8; // kCGScrollPhaseCancelled |
| if (JSStringIsEqualToUTF8CString(phaseStr, "maybegin")) |
| return 128; // kCGScrollPhaseMayBegin |
| |
| ASSERT_NOT_REACHED(); |
| return 0; |
| } |
| |
| static uint64_t cgEventMomentumPhaseFromString(JSStringRef phaseStr) |
| { |
| if (JSStringIsEqualToUTF8CString(phaseStr, "none")) |
| return 0; // kCGMomentumScrollPhaseNone |
| if (JSStringIsEqualToUTF8CString(phaseStr, "begin")) |
| return 1; // kCGMomentumScrollPhaseBegin |
| if (JSStringIsEqualToUTF8CString(phaseStr, "continue")) |
| return 2; // kCGMomentumScrollPhaseContinue |
| if (JSStringIsEqualToUTF8CString(phaseStr, "end")) |
| return 3; // kCGMomentumScrollPhaseEnd |
| |
| ASSERT_NOT_REACHED(); |
| return 0; |
| } |
| |
| void EventSendingController::mouseScrollByWithWheelAndMomentumPhases(int x, int y, JSStringRef phaseStr, JSStringRef momentumStr) |
| { |
| if (m_isDisabled) |
| return; |
| |
| uint64_t phase = cgEventPhaseFromString(phaseStr); |
| uint64_t momentum = cgEventMomentumPhaseFromString(momentumStr); |
| |
| if (phase == 4 /* kCGScrollPhaseEnded */ || phase == 8 /* kCGScrollPhaseCancelled */) |
| m_sentWheelPhaseEndOrCancel = true; |
| if (momentum == 3 /* kCGMomentumScrollPhaseEnd */) |
| m_sentWheelMomentumPhaseEnd = true; |
| |
| WKBundlePageForceRepaint(InjectedBundle::singleton().page()->page()); // Triggers a scrolling tree commit. |
| |
| auto body = createEventSenderDictionary("MouseScrollByWithWheelAndMomentumPhases"); |
| setValue(body, "X", adoptWK(WKDoubleCreate(x))); |
| setValue(body, "Y", adoptWK(WKDoubleCreate(y))); |
| setValue(body, "Phase", phase); |
| setValue(body, "Momentum", momentum); |
| postPageMessage("EventSender", body); |
| } |
| |
| void EventSendingController::setWheelHasPreciseDeltas(bool hasPreciseDeltas) |
| { |
| if (m_isDisabled) |
| return; |
| |
| auto body = createEventSenderDictionary("SetWheelHasPreciseDeltas"); |
| setValue(body, "HasPreciseDeltas", hasPreciseDeltas); |
| postPageMessage("EventSender", body); |
| } |
| |
| void EventSendingController::continuousMouseScrollBy(int x, int y, bool paged) |
| { |
| if (m_isDisabled) |
| return; |
| |
| auto body = createEventSenderDictionary("ContinuousMouseScrollBy"); |
| setValue(body, "X", adoptWK(WKDoubleCreate(x))); |
| setValue(body, "Y", adoptWK(WKDoubleCreate(y))); |
| setValue(body, "Paged", paged); |
| // FIXME: This message should be asynchronous, as scrolling is intrinsically asynchronous. |
| // See also: <https://bugs.webkit.org/show_bug.cgi?id=148256>. |
| postSynchronousPageMessage("EventSender", body); |
| } |
| |
| JSValueRef EventSendingController::contextClick(JSContextRef context) |
| { |
| #if ENABLE(CONTEXT_MENUS) |
| auto array = JSObjectMakeArray(context, 0, 0, 0); |
| if (m_isDisabled) |
| return array; |
| |
| auto page = InjectedBundle::singleton().page()->page(); |
| auto menuEntries = adoptWK(WKBundlePageCopyContextMenuAtPointInWindow(page, m_position)); |
| if (!menuEntries) |
| return array; |
| |
| size_t entriesSize = WKArrayGetSize(menuEntries.get()); |
| for (size_t i = 0; i < entriesSize; ++i) { |
| WKContextMenuItemRef item = dynamic_wk_cast<WKContextMenuItemRef>(WKArrayGetItemAtIndex(menuEntries.get(), i)); |
| MenuItemPrivateData* privateData = new MenuItemPrivateData(page, item); |
| JSObjectSetPropertyAtIndex(context, array, i, JSObjectMake(context, getMenuItemClass(), privateData), 0); |
| } |
| |
| return array; |
| #else |
| return JSValueMakeUndefined(context); |
| #endif |
| } |
| |
| void EventSendingController::textZoomIn() |
| { |
| if (m_isDisabled) |
| return; |
| |
| auto body = createEventSenderDictionary("SetTextZoom"); |
| setValue(body, "ZoomIn", true); |
| postSynchronousPageMessage("EventSender", body); |
| } |
| |
| void EventSendingController::textZoomOut() |
| { |
| if (m_isDisabled) |
| return; |
| |
| auto body = createEventSenderDictionary("SetTextZoom"); |
| setValue(body, "ZoomIn", false); |
| postSynchronousPageMessage("EventSender", body); |
| } |
| |
| void EventSendingController::zoomPageIn() |
| { |
| if (m_isDisabled) |
| return; |
| |
| auto body = createEventSenderDictionary("SetPageZoom"); |
| setValue(body, "ZoomIn", true); |
| postSynchronousPageMessage("EventSender", body); |
| } |
| |
| void EventSendingController::zoomPageOut() |
| { |
| if (m_isDisabled) |
| return; |
| |
| auto body = createEventSenderDictionary("SetPageZoom"); |
| setValue(body, "ZoomIn", false); |
| postSynchronousPageMessage("EventSender", body); |
| } |
| |
| MonitorWheelEventsOptions* toMonitorWheelEventsOptions(JSContextRef context, JSValueRef argument) |
| { |
| if (!JSValueIsObject(context, argument)) |
| return nullptr; |
| |
| static MonitorWheelEventsOptions options; |
| options.resetLatching = booleanProperty(context, (JSObjectRef)argument, "resetLatching", true); |
| return &options; |
| } |
| |
| void EventSendingController::monitorWheelEvents(MonitorWheelEventsOptions* options) |
| { |
| auto page = InjectedBundle::singleton().page()->page(); |
| |
| m_sentWheelPhaseEndOrCancel = false; |
| m_sentWheelMomentumPhaseEnd = false; |
| WKBundlePageStartMonitoringScrollOperations(page, options ? options->resetLatching : true); |
| } |
| |
| struct ScrollCompletionCallbackData { |
| WTF_DEPRECATED_MAKE_STRUCT_FAST_ALLOCATED(ScrollCompletionCallbackData); |
| JSContextRef m_context; |
| JSObjectRef m_function; |
| |
| ScrollCompletionCallbackData(JSContextRef context, JSObjectRef function) |
| : m_context(context), m_function(function) |
| { |
| } |
| }; |
| |
| static void executeCallback(void* context) |
| { |
| if (!context) |
| return; |
| |
| std::unique_ptr<ScrollCompletionCallbackData> callbackData(reinterpret_cast<ScrollCompletionCallbackData*>(context)); |
| |
| JSObjectCallAsFunction(callbackData->m_context, callbackData->m_function, nullptr, 0, nullptr, nullptr); |
| JSValueUnprotect(callbackData->m_context, callbackData->m_function); |
| } |
| |
| void EventSendingController::callAfterScrollingCompletes(JSContextRef context, JSValueRef functionCallback) |
| { |
| if (!functionCallback) |
| return; |
| |
| auto page = InjectedBundle::singleton().page()->page(); |
| |
| JSObjectRef functionCallbackObject = JSValueToObject(context, functionCallback, nullptr); |
| if (!functionCallbackObject) |
| return; |
| |
| JSValueProtect(context, functionCallbackObject); |
| |
| auto scrollCompletionCallbackData = makeUnique<ScrollCompletionCallbackData>(context, functionCallbackObject); |
| auto scrollCompletionCallbackDataPtr = scrollCompletionCallbackData.release(); |
| bool callbackWillBeCalled = WKBundlePageRegisterScrollOperationCompletionCallback(page, executeCallback, m_sentWheelPhaseEndOrCancel, m_sentWheelMomentumPhaseEnd, scrollCompletionCallbackDataPtr); |
| if (!callbackWillBeCalled) { |
| // Reassign raw pointer to std::unique_ptr<> so it will not be leaked. |
| scrollCompletionCallbackData.reset(scrollCompletionCallbackDataPtr); |
| } |
| } |
| |
| #if ENABLE(TOUCH_EVENTS) |
| |
| void EventSendingController::addTouchPoint(int x, int y) |
| { |
| if (m_isDisabled) |
| return; |
| |
| auto body = createEventSenderDictionary("AddTouchPoint"); |
| setValue(body, "X", static_cast<uint64_t>(x)); |
| setValue(body, "Y", static_cast<uint64_t>(y)); |
| postSynchronousPageMessage("EventSender", body); |
| } |
| |
| void EventSendingController::updateTouchPoint(int index, int x, int y) |
| { |
| if (m_isDisabled) |
| return; |
| |
| auto body = createEventSenderDictionary("UpdateTouchPoint"); |
| setValue(body, "Index", static_cast<uint64_t>(index)); |
| setValue(body, "X", static_cast<uint64_t>(x)); |
| setValue(body, "Y", static_cast<uint64_t>(y)); |
| postSynchronousPageMessage("EventSender", body); |
| } |
| |
| void EventSendingController::setTouchModifier(JSStringRef modifier, bool enable) |
| { |
| if (m_isDisabled) |
| return; |
| |
| auto body = createEventSenderDictionary("SetTouchModifier"); |
| setValue(body, "Modifier", parseTouchModifier(modifier)); |
| setValue(body, "Enable", enable); |
| postSynchronousPageMessage("EventSender", body); |
| } |
| |
| void EventSendingController::setTouchPointRadius(int radiusX, int radiusY) |
| { |
| if (m_isDisabled) |
| return; |
| |
| auto body = createEventSenderDictionary("SetTouchPointRadius"); |
| setValue(body, "RadiusX", static_cast<uint64_t>(radiusX)); |
| setValue(body, "RadiusY", static_cast<uint64_t>(radiusY)); |
| postSynchronousPageMessage("EventSender", body); |
| } |
| |
| void EventSendingController::touchStart() |
| { |
| if (m_isDisabled) |
| return; |
| |
| auto body = createEventSenderDictionary("TouchStart"); |
| postSynchronousPageMessage("EventSender", body); |
| } |
| |
| void EventSendingController::touchMove() |
| { |
| if (m_isDisabled) |
| return; |
| |
| auto body = createEventSenderDictionary("TouchMove"); |
| postSynchronousPageMessage("EventSender", body); |
| } |
| |
| void EventSendingController::touchEnd() |
| { |
| if (m_isDisabled) |
| return; |
| |
| auto body = createEventSenderDictionary("TouchEnd"); |
| postSynchronousPageMessage("EventSender", body); |
| } |
| |
| void EventSendingController::touchCancel() |
| { |
| if (m_isDisabled) |
| return; |
| |
| auto body = createEventSenderDictionary("TouchCancel"); |
| postSynchronousPageMessage("EventSender", body); |
| } |
| |
| void EventSendingController::clearTouchPoints() |
| { |
| if (m_isDisabled) |
| return; |
| |
| auto body = createEventSenderDictionary("ClearTouchPoints"); |
| postSynchronousPageMessage("EventSender", body); |
| } |
| |
| void EventSendingController::releaseTouchPoint(int index) |
| { |
| if (m_isDisabled) |
| return; |
| |
| auto body = createEventSenderDictionary("ReleaseTouchPoint"); |
| setValue(body, "Index", static_cast<uint64_t>(index)); |
| postSynchronousPageMessage("EventSender", body); |
| } |
| |
| void EventSendingController::cancelTouchPoint(int index) |
| { |
| if (m_isDisabled) |
| return; |
| |
| auto body = createEventSenderDictionary("CancelTouchPoint"); |
| setValue(body, "Index", static_cast<uint64_t>(index)); |
| postSynchronousPageMessage("EventSender", body); |
| } |
| |
| #endif |
| |
| void EventSendingController::smartMagnify() |
| { |
| if (m_isDisabled) |
| return; |
| |
| #if PLATFORM(MAC) |
| auto body = createEventSenderDictionary("SmartMagnify"); |
| postSynchronousPageMessage("EventSender", body); |
| #endif |
| } |
| |
| #if ENABLE(MAC_GESTURE_EVENTS) |
| |
| void EventSendingController::scaleGestureStart(double scale) |
| { |
| if (m_isDisabled) |
| return; |
| |
| auto body = createEventSenderDictionary("ScaleGestureStart"); |
| setValue(body, "Scale", scale); |
| postSynchronousPageMessage("EventSender", body); |
| } |
| |
| void EventSendingController::scaleGestureChange(double scale) |
| { |
| if (m_isDisabled) |
| return; |
| |
| auto body = createEventSenderDictionary("ScaleGestureChange"); |
| setValue(body, "Scale", scale); |
| postSynchronousPageMessage("EventSender", body); |
| } |
| |
| void EventSendingController::scaleGestureEnd(double scale) |
| { |
| if (m_isDisabled) |
| return; |
| |
| auto body = createEventSenderDictionary("ScaleGestureEnd"); |
| setValue(body, "Scale", scale); |
| postSynchronousPageMessage("EventSender", body); |
| } |
| |
| #endif // ENABLE(MAC_GESTURE_EVENTS) |
| |
| // Object Creation |
| |
| void EventSendingController::makeWindowObject(JSContextRef context) |
| { |
| setGlobalObjectProperty(context, "eventSender", this); |
| } |
| |
| } // namespace WTR |