blob: de530ee5ae8c65bdaa9c4fc41e62348e965b745e [file] [log] [blame]
/*
* Copyright (C) 2004-2016 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. ``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
* 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.
*/
#pragma once
#include "FloatPoint.h"
#include "IntPoint.h"
#include "PlatformEvent.h"
#include <wtf/WindowsExtras.h>
#if PLATFORM(GTK)
typedef struct _GdkEventScroll GdkEventScroll;
#endif
#if ENABLE(KINETIC_SCROLLING)
namespace WTF {
class TextStream;
}
#endif
namespace WebCore {
// The ScrollByPixelWheelEvent is a fine-grained event that specifies the precise number of pixels to scroll.
// It is sent directly by touch pads on macOS, or synthesized when platforms generate line-by-line scrolling events.
//
// The ScrollByPageWheelEvent indicates that the wheel event should scroll an entire page.
// In this case, WebKit built in paging behavior is used to page up and down.
// This yields the same behavior as clicking in a scrollbar track to page up and down.
enum PlatformWheelEventGranularity : uint8_t {
ScrollByPageWheelEvent,
ScrollByPixelWheelEvent,
};
#if ENABLE(KINETIC_SCROLLING)
enum PlatformWheelEventPhase : uint8_t {
PlatformWheelEventPhaseNone = 0,
PlatformWheelEventPhaseBegan = 1 << 0,
PlatformWheelEventPhaseStationary = 1 << 1,
PlatformWheelEventPhaseChanged = 1 << 2,
PlatformWheelEventPhaseEnded = 1 << 3,
PlatformWheelEventPhaseCancelled = 1 << 4,
PlatformWheelEventPhaseMayBegin = 1 << 5,
};
WTF::TextStream& operator<<(WTF::TextStream&, PlatformWheelEventPhase);
#endif
#if PLATFORM(WIN)
// How many pixels should we scroll per line? Gecko uses the height of the
// current line, which means scroll distance changes as you go through the
// page or go to different pages. IE 7 is ~50 px/line, although the value
// seems to vary slightly by page and zoom level. Since IE 7 has a
// smoothing algorithm on scrolling, it can get away with slightly larger
// scroll values without feeling jerky. Here we use 100 px per three lines
// (the default scroll amount on Windows is three lines per wheel tick).
const float cScrollbarPixelsPerLine = 100.0f / 3.0f;
#endif
class PlatformWheelEvent : public PlatformEvent {
public:
PlatformWheelEvent()
: PlatformEvent(PlatformEvent::Wheel)
{
}
PlatformWheelEvent(IntPoint position, IntPoint globalPosition, float deltaX, float deltaY, float wheelTicksX, float wheelTicksY, PlatformWheelEventGranularity granularity, bool shiftKey, bool ctrlKey, bool altKey, bool metaKey)
: PlatformEvent(PlatformEvent::Wheel, shiftKey, ctrlKey, altKey, metaKey, { })
, m_position(position)
, m_globalPosition(globalPosition)
, m_deltaX(deltaX)
, m_deltaY(deltaY)
, m_wheelTicksX(wheelTicksX)
, m_wheelTicksY(wheelTicksY)
, m_granularity(granularity)
{
}
PlatformWheelEvent copySwappingDirection() const
{
PlatformWheelEvent copy = *this;
std::swap(copy.m_deltaX, copy.m_deltaY);
std::swap(copy.m_wheelTicksX, copy.m_wheelTicksY);
return copy;
}
PlatformWheelEvent copyWithDeltasAndVelocity(float deltaX, float deltaY, const FloatSize& velocity) const
{
PlatformWheelEvent copy = *this;
copy.m_deltaX = deltaX;
copy.m_deltaY = deltaY;
copy.m_scrollingVelocity = velocity;
return copy;
}
const IntPoint& position() const { return m_position; } // PlatformWindow coordinates.
const IntPoint& globalPosition() const { return m_globalPosition; } // Screen coordinates.
float deltaX() const { return m_deltaX; }
float deltaY() const { return m_deltaY; }
FloatSize delta() const { return { m_deltaX, m_deltaY}; }
float wheelTicksX() const { return m_wheelTicksX; }
float wheelTicksY() const { return m_wheelTicksY; }
PlatformWheelEventGranularity granularity() const { return m_granularity; }
bool directionInvertedFromDevice() const { return m_directionInvertedFromDevice; }
const FloatSize& scrollingVelocity() const { return m_scrollingVelocity; }
#if PLATFORM(GTK)
explicit PlatformWheelEvent(GdkEventScroll*);
#endif
#if PLATFORM(COCOA)
bool hasPreciseScrollingDeltas() const { return m_hasPreciseScrollingDeltas; }
void setHasPreciseScrollingDeltas(bool hasPreciseScrollingDeltas) { m_hasPreciseScrollingDeltas = hasPreciseScrollingDeltas; }
unsigned scrollCount() const { return m_scrollCount; }
float unacceleratedScrollingDeltaX() const { return m_unacceleratedScrollingDeltaX; }
float unacceleratedScrollingDeltaY() const { return m_unacceleratedScrollingDeltaY; }
#endif
#if ENABLE(ASYNC_SCROLLING)
bool useLatchedEventElement() const;
bool shouldConsiderLatching() const;
bool shouldResetLatching() const;
bool isEndOfMomentumScroll() const;
#else
bool useLatchedEventElement() const { return false; }
#endif
#if ENABLE(KINETIC_SCROLLING)
PlatformWheelEventPhase phase() const { return m_phase; }
PlatformWheelEventPhase momentumPhase() const { return m_momentumPhase; }
bool isEndOfNonMomentumScroll() const;
bool isTransitioningToMomentumScroll() const;
FloatPoint swipeVelocity() const;
#endif
#if PLATFORM(WIN)
PlatformWheelEvent(HWND, WPARAM, LPARAM, bool isMouseHWheel);
#endif
protected:
IntPoint m_position;
IntPoint m_globalPosition;
float m_deltaX { 0 };
float m_deltaY { 0 };
float m_wheelTicksX { 0 };
float m_wheelTicksY { 0 };
PlatformWheelEventGranularity m_granularity { ScrollByPixelWheelEvent };
bool m_directionInvertedFromDevice { false };
// Scrolling velocity in pixels per second.
FloatSize m_scrollingVelocity;
#if ENABLE(KINETIC_SCROLLING)
PlatformWheelEventPhase m_phase { PlatformWheelEventPhaseNone };
PlatformWheelEventPhase m_momentumPhase { PlatformWheelEventPhaseNone };
#endif
#if PLATFORM(COCOA)
bool m_hasPreciseScrollingDeltas { false };
unsigned m_scrollCount { 0 };
float m_unacceleratedScrollingDeltaX { 0 };
float m_unacceleratedScrollingDeltaY { 0 };
#endif
};
#if ENABLE(ASYNC_SCROLLING)
inline bool PlatformWheelEvent::useLatchedEventElement() const
{
return m_phase == PlatformWheelEventPhaseBegan
|| m_phase == PlatformWheelEventPhaseChanged
|| m_momentumPhase == PlatformWheelEventPhaseBegan
|| m_momentumPhase == PlatformWheelEventPhaseChanged
|| (m_phase == PlatformWheelEventPhaseEnded && m_momentumPhase == PlatformWheelEventPhaseNone);
}
inline bool PlatformWheelEvent::shouldConsiderLatching() const
{
return m_phase == PlatformWheelEventPhaseBegan || m_phase == PlatformWheelEventPhaseMayBegin;
}
inline bool PlatformWheelEvent::shouldResetLatching() const
{
return m_phase == PlatformWheelEventPhaseCancelled || m_phase == PlatformWheelEventPhaseMayBegin || isEndOfMomentumScroll();
}
inline bool PlatformWheelEvent::isEndOfMomentumScroll() const
{
return m_phase == PlatformWheelEventPhaseNone && m_momentumPhase == PlatformWheelEventPhaseEnded;
}
#endif // ENABLE(ASYNC_SCROLLING)
#if ENABLE(KINETIC_SCROLLING)
inline bool PlatformWheelEvent::isEndOfNonMomentumScroll() const
{
return m_phase == PlatformWheelEventPhaseEnded && m_momentumPhase == PlatformWheelEventPhaseNone;
}
inline bool PlatformWheelEvent::isTransitioningToMomentumScroll() const
{
return m_phase == PlatformWheelEventPhaseNone && m_momentumPhase == PlatformWheelEventPhaseBegan;
}
inline FloatPoint PlatformWheelEvent::swipeVelocity() const
{
// The swiping velocity is stored in the deltas of the event declaring it.
return isTransitioningToMomentumScroll() ? FloatPoint(m_wheelTicksX, m_wheelTicksY) : FloatPoint();
}
#endif // ENABLE(KINETIC_SCROLLING)
WTF::TextStream& operator<<(WTF::TextStream&, const PlatformWheelEvent&);
} // namespace WebCore