diff options
Diffstat (limited to 'dom/events/WheelEvent.cpp')
-rw-r--r-- | dom/events/WheelEvent.cpp | 178 |
1 files changed, 178 insertions, 0 deletions
diff --git a/dom/events/WheelEvent.cpp b/dom/events/WheelEvent.cpp new file mode 100644 index 0000000000..6b58826627 --- /dev/null +++ b/dom/events/WheelEvent.cpp @@ -0,0 +1,178 @@ +/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- */ +/* vim: set ts=8 sts=2 et sw=2 tw=80: */ +/* This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this file, + * You can obtain one at http://mozilla.org/MPL/2.0/. */ + +#include "mozilla/dom/MouseEventBinding.h" +#include "mozilla/dom/WheelEvent.h" +#include "mozilla/MouseEvents.h" +#include "prtime.h" + +namespace mozilla::dom { + +WheelEvent::WheelEvent(EventTarget* aOwner, nsPresContext* aPresContext, + WidgetWheelEvent* aWheelEvent) + : MouseEvent(aOwner, aPresContext, + aWheelEvent + ? aWheelEvent + : new WidgetWheelEvent(false, eVoidEvent, nullptr)), + mAppUnitsPerDevPixel(0) { + if (aWheelEvent) { + mEventIsInternal = false; + // If the delta mode is pixel, the WidgetWheelEvent's delta values are in + // device pixels. However, JS contents need the delta values in CSS pixels. + // We should store the value of mAppUnitsPerDevPixel here because + // it might be changed by changing zoom or something. + if (aWheelEvent->mDeltaMode == WheelEvent_Binding::DOM_DELTA_PIXEL) { + mAppUnitsPerDevPixel = aPresContext->AppUnitsPerDevPixel(); + } + } else { + mEventIsInternal = true; + mEvent->mRefPoint = LayoutDeviceIntPoint(0, 0); + mEvent->AsWheelEvent()->mInputSource = + MouseEvent_Binding::MOZ_SOURCE_UNKNOWN; + } +} + +void WheelEvent::InitWheelEvent( + const nsAString& aType, bool aCanBubble, bool aCancelable, + nsGlobalWindowInner* aView, int32_t aDetail, int32_t aScreenX, + int32_t aScreenY, int32_t aClientX, int32_t aClientY, uint16_t aButton, + EventTarget* aRelatedTarget, const nsAString& aModifiersList, + double aDeltaX, double aDeltaY, double aDeltaZ, uint32_t aDeltaMode) { + NS_ENSURE_TRUE_VOID(!mEvent->mFlags.mIsBeingDispatched); + + MouseEvent::InitMouseEvent(aType, aCanBubble, aCancelable, aView, aDetail, + aScreenX, aScreenY, aClientX, aClientY, aButton, + aRelatedTarget, aModifiersList); + + WidgetWheelEvent* wheelEvent = mEvent->AsWheelEvent(); + // When specified by the caller (for JS-created events), don't mess with the + // delta mode. + wheelEvent->mDeltaModeCheckingState = + WidgetWheelEvent::DeltaModeCheckingState::Checked; + wheelEvent->mDeltaX = aDeltaX; + wheelEvent->mDeltaY = aDeltaY; + wheelEvent->mDeltaZ = aDeltaZ; + wheelEvent->mDeltaMode = aDeltaMode; + wheelEvent->mAllowToOverrideSystemScrollSpeed = false; +} + +int32_t WheelEvent::WheelDeltaX(CallerType aCallerType) { + WidgetWheelEvent* ev = mEvent->AsWheelEvent(); + if (ev->mWheelTicksX != 0.0) { + return int32_t(-ev->mWheelTicksX * kNativeTicksToWheelDelta); + } + if (IsTrusted()) { + // We always return pixels regardless of the checking-state. + double pixelDelta = + ev->mDeltaMode == WheelEvent_Binding::DOM_DELTA_PIXEL + ? CSSCoord(DevToCssPixels(ev->OverriddenDeltaX())) + : ev->OverriddenDeltaX() * + CSSPixel::FromAppUnits(ev->mScrollAmount.width).Rounded(); + return int32_t(-std::round(pixelDelta * kTrustedDeltaToWheelDelta)); + } + return int32_t(-std::round(DeltaX(aCallerType))); // This matches Safari. +} + +int32_t WheelEvent::WheelDeltaY(CallerType aCallerType) { + WidgetWheelEvent* ev = mEvent->AsWheelEvent(); + if (ev->mWheelTicksY != 0.0) { + return int32_t(-ev->mWheelTicksY * kNativeTicksToWheelDelta); + } + + if (IsTrusted()) { + double pixelDelta = + ev->mDeltaMode == WheelEvent_Binding::DOM_DELTA_PIXEL + ? CSSCoord(DevToCssPixels(ev->OverriddenDeltaY())) + : ev->OverriddenDeltaY() * + CSSPixel::FromAppUnits(ev->mScrollAmount.height).Rounded(); + return int32_t(-std::round(pixelDelta * kTrustedDeltaToWheelDelta)); + } + return int32_t(-std::round(DeltaY(aCallerType))); // This matches Safari. +} + +double WheelEvent::ToWebExposedDelta(WidgetWheelEvent& aWidgetEvent, + double aDelta, nscoord aLineOrPageAmount, + CallerType aCallerType) { + using DeltaModeCheckingState = WidgetWheelEvent::DeltaModeCheckingState; + if (aCallerType != CallerType::System) { + if (aWidgetEvent.mDeltaModeCheckingState == + DeltaModeCheckingState::Unknown) { + aWidgetEvent.mDeltaModeCheckingState = DeltaModeCheckingState::Unchecked; + } + if (aWidgetEvent.mDeltaModeCheckingState == + DeltaModeCheckingState::Unchecked && + aWidgetEvent.mDeltaMode == WheelEvent_Binding::DOM_DELTA_LINE) { + return aDelta * CSSPixel::FromAppUnits(aLineOrPageAmount).Rounded(); + } + } + return DevToCssPixels(aDelta); +} + +double WheelEvent::DeltaX(CallerType aCallerType) { + WidgetWheelEvent* ev = mEvent->AsWheelEvent(); + return ToWebExposedDelta(*ev, ev->OverriddenDeltaX(), ev->mScrollAmount.width, + aCallerType); +} + +double WheelEvent::DeltaY(CallerType aCallerType) { + WidgetWheelEvent* ev = mEvent->AsWheelEvent(); + return ToWebExposedDelta(*ev, ev->OverriddenDeltaY(), + ev->mScrollAmount.height, aCallerType); +} + +double WheelEvent::DeltaZ(CallerType aCallerType) { + WidgetWheelEvent* ev = mEvent->AsWheelEvent(); + // XXX Unclear what scroll amount we should use for deltaZ... + auto amount = std::max(ev->mScrollAmount.width, ev->mScrollAmount.height); + return ToWebExposedDelta(*ev, ev->mDeltaZ, amount, aCallerType); +} + +uint32_t WheelEvent::DeltaMode(CallerType aCallerType) { + using DeltaModeCheckingState = WidgetWheelEvent::DeltaModeCheckingState; + + WidgetWheelEvent* ev = mEvent->AsWheelEvent(); + uint32_t mode = ev->mDeltaMode; + if (aCallerType != CallerType::System) { + if (ev->mDeltaModeCheckingState == DeltaModeCheckingState::Unknown) { + ev->mDeltaModeCheckingState = DeltaModeCheckingState::Checked; + } else if (ev->mDeltaModeCheckingState == + DeltaModeCheckingState::Unchecked && + mode == WheelEvent_Binding::DOM_DELTA_LINE) { + return WheelEvent_Binding::DOM_DELTA_PIXEL; + } + } + + return mode; +} + +already_AddRefed<WheelEvent> WheelEvent::Constructor( + const GlobalObject& aGlobal, const nsAString& aType, + const WheelEventInit& aParam) { + nsCOMPtr<EventTarget> t = do_QueryInterface(aGlobal.GetAsSupports()); + RefPtr<WheelEvent> e = new WheelEvent(t, nullptr, nullptr); + bool trusted = e->Init(t); + e->InitWheelEvent(aType, aParam.mBubbles, aParam.mCancelable, aParam.mView, + aParam.mDetail, aParam.mScreenX, aParam.mScreenY, + aParam.mClientX, aParam.mClientY, aParam.mButton, + aParam.mRelatedTarget, u""_ns, aParam.mDeltaX, + aParam.mDeltaY, aParam.mDeltaZ, aParam.mDeltaMode); + e->InitializeExtraMouseEventDictionaryMembers(aParam); + e->SetTrusted(trusted); + e->SetComposed(aParam.mComposed); + return e.forget(); +} + +} // namespace mozilla::dom + +using namespace mozilla; +using namespace mozilla::dom; + +already_AddRefed<WheelEvent> NS_NewDOMWheelEvent(EventTarget* aOwner, + nsPresContext* aPresContext, + WidgetWheelEvent* aEvent) { + RefPtr<WheelEvent> it = new WheelEvent(aOwner, aPresContext, aEvent); + return it.forget(); +} |