mirror of
https://gitlab.winehq.org/wine/wine-gecko.git
synced 2024-09-13 09:24:08 -07:00
559 lines
21 KiB
C++
559 lines
21 KiB
C++
/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
|
|
/* vim: set sw=4 ts=8 et 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/. */
|
|
|
|
#ifndef mozilla_layers_AsyncPanZoomController_h
|
|
#define mozilla_layers_AsyncPanZoomController_h
|
|
|
|
#include "GeckoContentController.h"
|
|
#include "mozilla/Attributes.h"
|
|
#include "mozilla/Monitor.h"
|
|
#include "mozilla/RefPtr.h"
|
|
#include "mozilla/TimeStamp.h"
|
|
#include "InputData.h"
|
|
#include "Axis.h"
|
|
#include "nsContentUtils.h"
|
|
#include "TaskThrottler.h"
|
|
|
|
#include "base/message_loop.h"
|
|
|
|
namespace mozilla {
|
|
namespace layers {
|
|
|
|
class CompositorParent;
|
|
class GestureEventListener;
|
|
class ContainerLayer;
|
|
|
|
/**
|
|
* Controller for all panning and zooming logic. Any time a user input is
|
|
* detected and it must be processed in some way to affect what the user sees,
|
|
* it goes through here. Listens for any input event from InputData and can
|
|
* optionally handle nsGUIEvent-derived touch events, but this must be done on
|
|
* the main thread. Note that this class completely cross-platform.
|
|
*
|
|
* Input events originate on the UI thread of the platform that this runs on,
|
|
* and are then sent to this class. This class processes the event in some way;
|
|
* for example, a touch move will usually lead to a panning of content (though
|
|
* of course there are exceptions, such as if content preventDefaults the event,
|
|
* or if the target frame is not scrollable). The compositor interacts with this
|
|
* class by locking it and querying it for the current transform matrix based on
|
|
* the panning and zooming logic that was invoked on the UI thread.
|
|
*
|
|
* Currently, each outer DOM window (i.e. a website in a tab, but not any
|
|
* subframes) has its own AsyncPanZoomController. In the future, to support
|
|
* asynchronously scrolled subframes, we want to have one AsyncPanZoomController
|
|
* per frame.
|
|
*/
|
|
class AsyncPanZoomController MOZ_FINAL {
|
|
NS_INLINE_DECL_THREADSAFE_REFCOUNTING(AsyncPanZoomController)
|
|
|
|
typedef mozilla::MonitorAutoLock MonitorAutoLock;
|
|
|
|
public:
|
|
enum GestureBehavior {
|
|
// The platform code is responsible for forwarding gesture events here. We
|
|
// will not attempt to generate gesture events from MultiTouchInputs.
|
|
DEFAULT_GESTURES,
|
|
// An instance of GestureEventListener is used to detect gestures. This is
|
|
// handled completely internally within this class.
|
|
USE_GESTURE_DETECTOR
|
|
};
|
|
|
|
/**
|
|
* Constant describing the tolerance in distance we use, multiplied by the
|
|
* device DPI, before we start panning the screen. This is to prevent us from
|
|
* accidentally processing taps as touch moves, and from very short/accidental
|
|
* touches moving the screen.
|
|
*/
|
|
static const float TOUCH_START_TOLERANCE;
|
|
|
|
AsyncPanZoomController(GeckoContentController* aController,
|
|
GestureBehavior aGestures = DEFAULT_GESTURES);
|
|
~AsyncPanZoomController();
|
|
|
|
// --------------------------------------------------------------------------
|
|
// These methods must only be called on the controller/UI thread.
|
|
//
|
|
|
|
/**
|
|
* General handler for incoming input events. Manipulates the frame metrics
|
|
* basde on what type of input it is. For example, a PinchGestureEvent will
|
|
* cause scaling. This should only be called externally to this class.
|
|
* HandleInputEvent() should be used internally.
|
|
*/
|
|
nsEventStatus ReceiveInputEvent(const InputData& aEvent);
|
|
|
|
/**
|
|
* Special handler for nsInputEvents. Also sets |aOutEvent| (which is assumed
|
|
* to be an already-existing instance of an nsInputEvent which may be an
|
|
* nsTouchEvent) to have its touch points in DOM space. This is so that the
|
|
* touches can be passed through the DOM and content can handle them.
|
|
*
|
|
* NOTE: Be careful of invoking the nsInputEvent variant. This can only be
|
|
* called on the main thread. See widget/InputData.h for more information on
|
|
* why we have InputData and nsInputEvent separated.
|
|
*/
|
|
nsEventStatus ReceiveInputEvent(const nsInputEvent& aEvent,
|
|
nsInputEvent* aOutEvent);
|
|
|
|
/**
|
|
* Updates the composition bounds, i.e. the dimensions of the final size of
|
|
* the frame this is tied to during composition onto, in device pixels. In
|
|
* general, this will just be:
|
|
* { x = 0, y = 0, width = surface.width, height = surface.height }, however
|
|
* there is no hard requirement for this.
|
|
*/
|
|
void UpdateCompositionBounds(const nsIntRect& aCompositionBounds);
|
|
|
|
/**
|
|
* We have found a scrollable subframe, so disable our machinery until we hit
|
|
* a touch end or a new touch start. This prevents us from accidentally
|
|
* panning both the subframe and the parent frame.
|
|
*
|
|
* XXX/bug 775452: We should eventually be supporting async scrollable
|
|
* subframes.
|
|
*/
|
|
void CancelDefaultPanZoom();
|
|
|
|
/**
|
|
* Kicks an animation to zoom to a rect. This may be either a zoom out or zoom
|
|
* in. The actual animation is done on the compositor thread after being set
|
|
* up. |aRect| must be given in CSS pixels, relative to the document.
|
|
*/
|
|
void ZoomToRect(const gfxRect& aRect);
|
|
|
|
/**
|
|
* If we have touch listeners, this should always be called when we know
|
|
* definitively whether or not content has preventDefaulted any touch events
|
|
* that have come in. If |aPreventDefault| is true, any touch events in the
|
|
* queue will be discarded.
|
|
*/
|
|
void ContentReceivedTouch(bool aPreventDefault);
|
|
|
|
/**
|
|
* Updates any zoom constraints contained in the <meta name="viewport"> tag.
|
|
* We try to obey everything it asks us elsewhere, but here we only handle
|
|
* minimum-scale, maximum-scale, and user-scalable.
|
|
*/
|
|
void UpdateZoomConstraints(bool aAllowZoom, float aMinScale, float aMaxScale);
|
|
|
|
// --------------------------------------------------------------------------
|
|
// These methods must only be called on the compositor thread.
|
|
//
|
|
|
|
/**
|
|
* The compositor calls this when it's about to draw pannable/zoomable content
|
|
* and is setting up transforms for compositing the layer tree. This is not
|
|
* idempotent. For example, a fling transform can be applied each time this is
|
|
* called (though not necessarily). |aSampleTime| is the time that this is
|
|
* sampled at; this is used for interpolating animations. Calling this sets a
|
|
* new transform in |aNewTransform| which should be applied directly to the
|
|
* shadow layer of the frame (do not multiply it in as the code already does
|
|
* this internally with |aLayer|'s transform).
|
|
*
|
|
* Return value indicates whether or not any currently running animation
|
|
* should continue. That is, if true, the compositor should schedule another
|
|
* composite.
|
|
*/
|
|
bool SampleContentTransformForFrame(const TimeStamp& aSampleTime,
|
|
ContainerLayer* aLayer,
|
|
gfx3DMatrix* aNewTransform);
|
|
|
|
/**
|
|
* A shadow layer update has arrived. |aViewportFrame| is the new FrameMetrics
|
|
* for the top-level frame. |aIsFirstPaint| is a flag passed from the shadow
|
|
* layers code indicating that the frame metrics being sent with this call are
|
|
* the initial metrics and the initial paint of the frame has just happened.
|
|
*/
|
|
void NotifyLayersUpdated(const FrameMetrics& aViewportFrame, bool aIsFirstPaint);
|
|
|
|
/**
|
|
* The platform implementation must set the compositor parent so that we can
|
|
* request composites.
|
|
*/
|
|
void SetCompositorParent(CompositorParent* aCompositorParent);
|
|
|
|
// --------------------------------------------------------------------------
|
|
// These methods can be called from any thread.
|
|
//
|
|
|
|
/**
|
|
* Sets the CSS page rect, and calculates a new page rect based on the zoom
|
|
* level of the current metrics and the passed in CSS page rect.
|
|
*/
|
|
void SetPageRect(const gfx::Rect& aCSSPageRect);
|
|
|
|
/**
|
|
* Sets the DPI of the device for use within panning and zooming logic. It is
|
|
* a platform responsibility to set this on initialization of this class and
|
|
* whenever it changes.
|
|
*/
|
|
void SetDPI(int aDPI);
|
|
|
|
/**
|
|
* Gets the DPI of the device for use outside the panning and zooming logic.
|
|
* It defaults to 72 if not set using SetDPI() at any point.
|
|
*/
|
|
int GetDPI();
|
|
|
|
/**
|
|
* Recalculates the displayport. Ideally, this should paint an area bigger
|
|
* than the composite-to dimensions so that when you scroll down, you don't
|
|
* checkerboard immediately. This includes a bunch of logic, including
|
|
* algorithms to bias painting in the direction of the velocity.
|
|
*/
|
|
static const gfx::Rect CalculatePendingDisplayPort(
|
|
const FrameMetrics& aFrameMetrics,
|
|
const gfx::Point& aVelocity,
|
|
const gfx::Point& aAcceleration,
|
|
double aEstimatedPaintDuration);
|
|
|
|
/**
|
|
* Return the scale factor needed to fit the viewport in |aMetrics|
|
|
* into its composition bounds.
|
|
*/
|
|
static gfxSize CalculateIntrinsicScale(const FrameMetrics& aMetrics);
|
|
|
|
/**
|
|
* Return the resolution that content should be rendered at given
|
|
* the configuration in aFrameMetrics: viewport dimensions, zoom
|
|
* factor, etc. (The mResolution member of aFrameMetrics is
|
|
* ignored.)
|
|
*/
|
|
static gfxSize CalculateResolution(const FrameMetrics& aMetrics);
|
|
|
|
static gfx::Rect CalculateCompositedRectInCssPixels(const FrameMetrics& aMetrics);
|
|
|
|
protected:
|
|
/**
|
|
* Internal handler for ReceiveInputEvent(). Does all the actual work.
|
|
*/
|
|
nsEventStatus HandleInputEvent(const InputData& aEvent);
|
|
|
|
/**
|
|
* Helper method for touches beginning. Sets everything up for panning and any
|
|
* multitouch gestures.
|
|
*/
|
|
nsEventStatus OnTouchStart(const MultiTouchInput& aEvent);
|
|
|
|
/**
|
|
* Helper method for touches moving. Does any transforms needed when panning.
|
|
*/
|
|
nsEventStatus OnTouchMove(const MultiTouchInput& aEvent);
|
|
|
|
/**
|
|
* Helper method for touches ending. Redraws the screen if necessary and does
|
|
* any cleanup after a touch has ended.
|
|
*/
|
|
nsEventStatus OnTouchEnd(const MultiTouchInput& aEvent);
|
|
|
|
/**
|
|
* Helper method for touches being cancelled. Treated roughly the same as a
|
|
* touch ending (OnTouchEnd()).
|
|
*/
|
|
nsEventStatus OnTouchCancel(const MultiTouchInput& aEvent);
|
|
|
|
/**
|
|
* Helper method for scales beginning. Distinct from the OnTouch* handlers in
|
|
* that this implies some outside implementation has determined that the user
|
|
* is pinching.
|
|
*/
|
|
nsEventStatus OnScaleBegin(const PinchGestureInput& aEvent);
|
|
|
|
/**
|
|
* Helper method for scaling. As the user moves their fingers when pinching,
|
|
* this changes the scale of the page.
|
|
*/
|
|
nsEventStatus OnScale(const PinchGestureInput& aEvent);
|
|
|
|
/**
|
|
* Helper method for scales ending. Redraws the screen if necessary and does
|
|
* any cleanup after a scale has ended.
|
|
*/
|
|
nsEventStatus OnScaleEnd(const PinchGestureInput& aEvent);
|
|
|
|
/**
|
|
* Helper method for long press gestures.
|
|
*
|
|
* XXX: Implement this.
|
|
*/
|
|
nsEventStatus OnLongPress(const TapGestureInput& aEvent);
|
|
|
|
/**
|
|
* Helper method for single tap gestures.
|
|
*
|
|
* XXX: Implement this.
|
|
*/
|
|
nsEventStatus OnSingleTapUp(const TapGestureInput& aEvent);
|
|
|
|
/**
|
|
* Helper method for a single tap confirmed.
|
|
*
|
|
* XXX: Implement this.
|
|
*/
|
|
nsEventStatus OnSingleTapConfirmed(const TapGestureInput& aEvent);
|
|
|
|
/**
|
|
* Helper method for double taps.
|
|
*
|
|
* XXX: Implement this.
|
|
*/
|
|
nsEventStatus OnDoubleTap(const TapGestureInput& aEvent);
|
|
|
|
/**
|
|
* Helper method to cancel any gesture currently going to Gecko. Used
|
|
* primarily when a user taps the screen over some clickable content but then
|
|
* pans down instead of letting go (i.e. to cancel a previous touch so that a
|
|
* new one can properly take effect.
|
|
*/
|
|
nsEventStatus OnCancelTap(const TapGestureInput& aEvent);
|
|
|
|
/**
|
|
* Scrolls the viewport by an X,Y offset.
|
|
*/
|
|
void ScrollBy(const gfx::Point& aOffset);
|
|
|
|
/**
|
|
* Scales the viewport by an amount (note that it multiplies this scale in to
|
|
* the current scale, it doesn't set it to |aScale|). Also considers a focus
|
|
* point so that the page zooms outward from that point.
|
|
*
|
|
* XXX: Fix focus point calculations.
|
|
*/
|
|
void ScaleWithFocus(float aScale, const nsIntPoint& aFocus);
|
|
|
|
/**
|
|
* Schedules a composite on the compositor thread. Wrapper for
|
|
* CompositorParent::ScheduleRenderOnCompositorThread().
|
|
*/
|
|
void ScheduleComposite();
|
|
|
|
/**
|
|
* Cancels any currently running animation. Note that all this does is set the
|
|
* state of the AsyncPanZoomController back to NOTHING, but it is the
|
|
* animation's responsibility to check this before advancing.
|
|
*
|
|
* *** The monitor must be held while calling this.
|
|
*/
|
|
void CancelAnimation();
|
|
|
|
/**
|
|
* Gets the displacement of the current touch since it began. That is, it is
|
|
* the distance between the current position and the initial position of the
|
|
* current touch (this only makes sense if a touch is currently happening and
|
|
* OnTouchMove() is being invoked).
|
|
*/
|
|
float PanDistance();
|
|
|
|
/**
|
|
* Gets a vector of the velocities of each axis.
|
|
*/
|
|
const gfx::Point GetVelocityVector();
|
|
|
|
/**
|
|
* Gets a vector of the acceleration factors of each axis.
|
|
*/
|
|
const gfx::Point GetAccelerationVector();
|
|
|
|
/**
|
|
* Gets a reference to the first SingleTouchData from a MultiTouchInput. This
|
|
* gets only the first one and assumes the rest are either missing or not
|
|
* relevant.
|
|
*/
|
|
SingleTouchData& GetFirstSingleTouch(const MultiTouchInput& aEvent);
|
|
|
|
/**
|
|
* Sets up anything needed for panning. This takes us out of the "TOUCHING"
|
|
* state and starts actually panning us.
|
|
*/
|
|
void StartPanning(const MultiTouchInput& aStartPoint);
|
|
|
|
/**
|
|
* Wrapper for Axis::UpdateWithTouchAtDevicePoint(). Calls this function for
|
|
* both axes and factors in the time delta from the last update.
|
|
*/
|
|
void UpdateWithTouchAtDevicePoint(const MultiTouchInput& aEvent);
|
|
|
|
/**
|
|
* Does any panning required due to a new touch event.
|
|
*/
|
|
void TrackTouch(const MultiTouchInput& aEvent);
|
|
|
|
/**
|
|
* Attempts to enlarge the displayport along a single axis. Returns whether or
|
|
* not the displayport was enlarged. This will fail in circumstances where the
|
|
* velocity along that axis is not high enough to need any changes. The
|
|
* displayport metrics are expected to be passed into |aDisplayPortOffset| and
|
|
* |aDisplayPortLength|. If enlarged, these will be updated with the new
|
|
* metrics.
|
|
*/
|
|
static bool EnlargeDisplayPortAlongAxis(float aSkateSizeMultiplier,
|
|
double aEstimatedPaintDuration,
|
|
float aCompositionBounds,
|
|
float aVelocity,
|
|
float aAcceleration,
|
|
float* aDisplayPortOffset,
|
|
float* aDisplayPortLength);
|
|
|
|
/**
|
|
* Utility function to send updated FrameMetrics to Gecko so that it can paint
|
|
* the displayport area. Calls into GeckoContentController to do the actual
|
|
* work. Note that only one paint request can be active at a time. If a paint
|
|
* request is made while a paint is currently happening, it gets queued up. If
|
|
* a new paint request arrives before a paint is completed, the old request
|
|
* gets discarded.
|
|
*/
|
|
void RequestContentRepaint();
|
|
|
|
/**
|
|
* Advances a fling by an interpolated amount based on the passed in |aDelta|.
|
|
* This should be called whenever sampling the content transform for this
|
|
* frame. Returns true if the fling animation should be advanced by one frame,
|
|
* or false if there is no fling or the fling has ended.
|
|
*/
|
|
bool DoFling(const TimeDuration& aDelta);
|
|
|
|
/**
|
|
* Gets the current frame metrics. This is *not* the Gecko copy stored in the
|
|
* layers code.
|
|
*/
|
|
const FrameMetrics& GetFrameMetrics();
|
|
|
|
/**
|
|
* Timeout function for touch listeners. This should be called on a timer
|
|
* after we get our first touch event in a batch, under the condition that we
|
|
* have touch listeners. If a notification comes indicating whether or not
|
|
* content preventDefaulted a series of touch events before the timeout, the
|
|
* timeout should be cancelled.
|
|
*/
|
|
void TimeoutTouchListeners();
|
|
|
|
/**
|
|
* Utility function that sets the zoom and resolution simultaneously. This is
|
|
* useful when we want to repaint at the current zoom level.
|
|
*
|
|
* *** The monitor must be held while calling this.
|
|
*/
|
|
void SetZoomAndResolution(float aScale);
|
|
|
|
private:
|
|
enum PanZoomState {
|
|
NOTHING, /* no touch-start events received */
|
|
FLING, /* all touches removed, but we're still scrolling page */
|
|
TOUCHING, /* one touch-start event received */
|
|
PANNING, /* panning the frame */
|
|
PINCHING, /* nth touch-start, where n > 1. this mode allows pan and zoom */
|
|
ANIMATING_ZOOM, /* animated zoom to a new rect */
|
|
WAITING_LISTENERS, /* a state halfway between NOTHING and TOUCHING - the user has
|
|
put a finger down, but we don't yet know if a touch listener has
|
|
prevented the default actions yet. we still need to abort animations. */
|
|
};
|
|
|
|
/**
|
|
* Helper to set the current state. Holds the monitor before actually setting
|
|
* it. If the monitor is already held by the current thread, it is safe to
|
|
* instead use: |mState = NEWSTATE;|
|
|
*/
|
|
void SetState(PanZoomState aState);
|
|
|
|
nsRefPtr<CompositorParent> mCompositorParent;
|
|
TaskThrottler mPaintThrottler;
|
|
nsRefPtr<GeckoContentController> mGeckoContentController;
|
|
nsRefPtr<GestureEventListener> mGestureEventListener;
|
|
|
|
// Both |mFrameMetrics| and |mLastContentPaintMetrics| are protected by the
|
|
// monitor. Do not read from or modify either of them without locking.
|
|
FrameMetrics mFrameMetrics;
|
|
// These are the metrics at last content paint, the most recent
|
|
// values we were notified of in NotifyLayersUpdate().
|
|
FrameMetrics mLastContentPaintMetrics;
|
|
// The last metrics that we requested a paint for. These are used to make sure
|
|
// that we're not requesting a paint of the same thing that's already drawn.
|
|
// If we don't do this check, we don't get a ShadowLayersUpdated back.
|
|
FrameMetrics mLastPaintRequestMetrics;
|
|
|
|
// Old metrics from before we started a zoom animation. This is only valid
|
|
// when we are in the "ANIMATED_ZOOM" state. This is used so that we can
|
|
// interpolate between the start and end frames. We only use the
|
|
// |mViewportScrollOffset| and |mResolution| fields on this.
|
|
FrameMetrics mStartZoomToMetrics;
|
|
// Target metrics for a zoom to animation. This is only valid when we are in
|
|
// the "ANIMATED_ZOOM" state. We only use the |mViewportScrollOffset| and
|
|
// |mResolution| fields on this.
|
|
FrameMetrics mEndZoomToMetrics;
|
|
|
|
nsTArray<MultiTouchInput> mTouchQueue;
|
|
|
|
CancelableTask* mTouchListenerTimeoutTask;
|
|
|
|
AxisX mX;
|
|
AxisY mY;
|
|
|
|
// Most up-to-date constraints on zooming. These should always be reasonable
|
|
// values; for example, allowing a min zoom of 0.0 can cause very bad things
|
|
// to happen.
|
|
bool mAllowZoom;
|
|
float mMinZoom;
|
|
float mMaxZoom;
|
|
|
|
// Protects |mFrameMetrics|, |mLastContentPaintMetrics|, |mState| and
|
|
// |mMetaViewportInfo|. Before manipulating |mFrameMetrics| or
|
|
// |mLastContentPaintMetrics|, the monitor should be held. When setting
|
|
// |mState|, either the SetState() function can be used, or the monitor can be
|
|
// held and then |mState| updated. |mMetaViewportInfo| should be updated
|
|
// using UpdateMetaViewport().
|
|
Monitor mMonitor;
|
|
|
|
// The last time the compositor has sampled the content transform for this
|
|
// frame.
|
|
TimeStamp mLastSampleTime;
|
|
// The last time a touch event came through on the UI thread.
|
|
int32_t mLastEventTime;
|
|
|
|
// Start time of an animation. This is used for a zoom to animation to mark
|
|
// the beginning.
|
|
TimeStamp mAnimationStartTime;
|
|
|
|
// Stores the previous focus point if there is a pinch gesture happening. Used
|
|
// to allow panning by moving multiple fingers (thus moving the focus point).
|
|
nsIntPoint mLastZoomFocus;
|
|
|
|
// Stores the state of panning and zooming this frame. This is protected by
|
|
// |mMonitor|; that is, it should be held whenever this is updated.
|
|
PanZoomState mState;
|
|
|
|
// How long it took in the past to paint after a series of previous requests.
|
|
nsTArray<TimeDuration> mPreviousPaintDurations;
|
|
|
|
// When the last paint request started. Used to determine the duration of
|
|
// previous paints.
|
|
TimeStamp mPreviousPaintStartTime;
|
|
|
|
int mDPI;
|
|
|
|
// Stores the current paint status of the frame that we're managing. Repaints
|
|
// may be triggered by other things (like content doing things), in which case
|
|
// this status will not be updated. It is only changed when this class
|
|
// requests a repaint.
|
|
bool mWaitingForContentToPaint;
|
|
|
|
// Flag used to determine whether or not we should disable handling of the
|
|
// next batch of touch events. This is used for sync scrolling of subframes.
|
|
bool mDisableNextTouchBatch;
|
|
|
|
// Flag used to determine whether or not we should try to enter the
|
|
// WAITING_LISTENERS state. This is used in the case that we are processing a
|
|
// queued up event block. If set, this means that we are handling this queue
|
|
// and we don't want to queue the events back up again.
|
|
bool mHandlingTouchQueue;
|
|
|
|
friend class Axis;
|
|
};
|
|
|
|
}
|
|
}
|
|
|
|
#endif // mozilla_layers_PanZoomController_h
|