mirror of
https://github.com/roytam1/palemoon27.git
synced 2026-05-26 14:30:27 +00:00
5a5b052a7c
- Bug 930218 part 1. Factor out the computation of block-size taken up by box-sizing into a separate function so we can reuse it. r=dbaron (22033f4184) - Bug 930218 part 2. Account for the parent's box-sizing whe figuring out the percentage height of a kid with an intrinsic ratio for purposes of determining the parent's shrink-wrap width. r=dbaron (041711f58f) - Bug 1235306 - Fix -Wimplicit-fallthrough warnings in layout/. r=dholbert (581c212254) - Bug 1074971 - Add support for reserved commandkey combinations that can't be handled by content (e10s-only feature). r=smaug (e43d39dcef) - Bug 1186799 part.1 nsHTMLEditorEventListener should commit composition when it receives unacceptable mousedown event r=smaug (17f2c48e96) - Bug 1230216 - Changing nsIDOM*Event interfaces so that they don't inherit nsIDOMEvent. r=smaug (df77f91ecc) - fix cancelBubble situation and align to gecko (bc133dc50a) - Bug 1211402. Re-enable the upload step, but take out the stuff that no longer works (as in, everything.) r=me (cb7abd67d2) - align (bac4aec6cc) - Bug 1243608: P1. Only use video time if video frame contains seek target. r=cpearce (21d9e988ed) - Bug 1244477: Offset seek time by start time. r=jwwang (743c24b2f7) - Bug 1243608: P2. Pass the full SeekTarget object to MediaDecoderReader::Seek. r=cpearce (6ff30b4b65) - Bug 1243608: P3. Make SeekTarget::mTime a TimeUnit object. r=cpearce (357d9864de) - Bug 1243608: P4. Have MediaDecoderReader::SeekPromise return a TimeUnit. r=cpearce (53b476c62d) - Bug 1159343: Interrupt seek early when possible. r=jwwang (9e26e69593) - Bug 1243608: P5. Add type utility methods to SeekTarget class. r=cpearce (8e740bab50) - Bug 1243608: P6. Only seek audio to video seek time when performing a fast seek. r=cpearce (d2a51a71f0) - Bug 1233650. Part 1 - extract OutputStreamManager to its own file. r=roc. (52d533f923) - Bug 1230882. Part 2 - remove DecodedStream::BeginShutdown() and other unused code. r=roc. (b2820b8f14) - Bug 1231091. Part 1 - Add mVideoCompleted so MDSM can check when audio/video is done rendering. This removes the only caller of DecodedStream::IsFinished(). r=roc. (536f63e385) - Bug 1146086: Properly marking overridden member with override keyword. v2. a=bustage (b6ed1b4e6c) - Bug 1177243 - Use PodZero rather than memset in WebM decoders. r=rillian (f9853b72ee) - Bug 1230054: Remove unused WebMReader. r=kinetik (1f88fe3c43) - Bug 1230054: Add missing headers ON A CLOSED TREE. r=me (6e33accf03) - Bug 1231091. Part 2 - return correct promises when audio/video track is asked. r=roc. (aef6342e59) - Bug 1231091. Part 3 - resolve the end promise when all frames are rendered. r=roc. (bbb180dd90) - Bug 1231091. Part 4 - ensure the end promise is resolved in the special case where video duration is 0. r=roc. (7835c31e75) - Bug 1231091. Part 5 - Remove DecodedStream::IsFinished() and unused code. r=roc. (6dd02e5d6b) - Bug 1231091. Part 6 - fix test_streams_element_capture.html timeout. r=roc. (8aae5c09cc) - Bug 1232520 - dont' invoke AbstractThread::MainThread()->Dispatch() to avoid reentrant of AutoTaskDispatcher during tail dispatching phase. r=jya. (5434d0f370) - Bug 1233650. Part 2 - make OutputStreamManager ref-counted so it can be shared between MDSM and DecodedStream. r=roc. (93e7d0e065) - Bug 1233650. Part 3 - move creation of OutputStreamManager from DecodedStream to MDSM. r=roc. (a469dbcc9f) - Bug 1233650. Part 4 - remove unused functions from DecodedStream. r=roc. (ec8753365e) - Bug 1248314. Part 1 - Since OutputStreamManager::Connect/Disconnect is tightly coupled with the constructor/destructor of DecodedStreamData, it would improve cohesion to let DecodedStreamData manage an OutputStreamManager and know when to call OutputStreamManager::Connect/Disconnect. r=roc. (467d1472ca) - Bug 1234424. Part 1 - share the underlying value of MDSM::mSameOriginMedia with DecodedStream. r=roc. (ff0abefb26) - Bug 1234424. Part 2 - remove unused code. r=roc. (5a8266779b) - Bug 1248314. part 2 - move track initialization code into the constructor of DecodedStreamData. r=roc. (4d57f47654) - Bug 1248229. Part 1 - add test case to test if playback can work correctly after GC. r=roc. (83c81dc7cc) - Bug 1248229. Part 2 - GC might happen in between OutputStreamManager::Disconnect() and OutputStreamManager::Connect(). We need to check if the stream is already destroyed before trying to connect it. r=roc. (c5a0ed670c) - remove PM leftover (ed9ce00aad) - Bug 1236703: P1. Add debugging information for MSE to about:media plugin. r=kentuckyfriedtakahe (4385a86197) - Bug 1236703: P2. Add methods to retrieve debugging data on plain readers. r=jwwang (34bf637124) - Bug 1236703: P3. Add moz specific method to retrieve debug data to media object IDL. r=bz (65c95eff74) - Bug 1194721: Add additional PDU pack and unpack functions, r=shuang (d6ae416a10) - Bug 1261307: Convert Unix socket IPC code to |UniquePtr|, r=nfroyd (cd797f4581) - Bug 1236574 - use UniquePtr<T[]> instead of nsAutoArrayPtr<T> in SocketBase subclasses; r=tzimmerman (6a20e9e905) - Bug 1239207 - Don't process IPDL when not compiling; r=glandium (20ba7fb311) - Bug 1210099 - Use diagnostic assert for union discriminator checks (r=jld) (74844eee1b) - Bug 1208226 - Don't crash when failing to map a segment of shared memory. r=sotaro, billm (1822a634d9) - Bug 1236635 - Fix compile error in IPC unit tests (r=jld) (c82e0bfe61) - Bug 1263429 - Don't build libevent with sysctl on Linux. r=billm (00f248f61e) - Bug 1304266 - Remove libevent workaround for MacOS 10.4 bug (r=dvander) a=jcristau (676758a926) - Bug 1241776 - turn ENABLE_SHARED_ARRAY_BUFFER on for all channels. r=nbp (dc53fff545) - Bug 1248851 part 1 - Explicitly mark some release() calls result-unused. r=Waldo (b377e3d86e) - Bug 1246697 - Use simpler semantics for PersistentRooted<Traceable>; r=sfink (7afab5c807) - bug 1225649 use CreatePlanarYCbCrImage() now that CreateImage() is gone r=dvander (b8a30f3d84) - Bug 1234092: P1. Remove GStreamer support. r=kentuckyfriedtakahe (bf6a5d8811) - Bug 1234092: P2. Remove GStreamer check from configure. r=glandium (73b69b41db) - remove configure options (f9585af0d1) - Bug 1168309 - Add the define, MOZ_GONK_MEDIACODEC, to dom/media. r=cpearce. (a4481af932) - Bug 1240616 - mach mochitest-run suggests deprecated commands as an alternative r=cmanchester (17687db974) - Bug 1249838 - Avoid dependency from the mozconfig loader on mach. r=gps (a88506c027) - Bug 382721 - Part 0: Add missing includes and namespaces. r=jrmuizel (eecf253c5e)
777 lines
21 KiB
C++
777 lines
21 KiB
C++
/* -*- 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 "AccessibleCaretEventHub.h"
|
|
|
|
#include "AccessibleCaretLogger.h"
|
|
#include "AccessibleCaretManager.h"
|
|
#include "Layers.h"
|
|
#include "gfxPrefs.h"
|
|
#include "mozilla/MouseEvents.h"
|
|
#include "mozilla/TextEvents.h"
|
|
#include "mozilla/TouchEvents.h"
|
|
#include "mozilla/Preferences.h"
|
|
#include "nsCanvasFrame.h"
|
|
#include "nsDocShell.h"
|
|
#include "nsFocusManager.h"
|
|
#include "nsFrameSelection.h"
|
|
#include "nsITimer.h"
|
|
#include "nsPresContext.h"
|
|
|
|
namespace mozilla {
|
|
|
|
#undef AC_LOG
|
|
#define AC_LOG(message, ...) \
|
|
AC_LOG_BASE("AccessibleCaretEventHub (%p): " message, this, ##__VA_ARGS__);
|
|
|
|
#undef AC_LOGV
|
|
#define AC_LOGV(message, ...) \
|
|
AC_LOGV_BASE("AccessibleCaretEventHub (%p): " message, this, ##__VA_ARGS__);
|
|
|
|
NS_IMPL_ISUPPORTS(AccessibleCaretEventHub, nsIReflowObserver, nsIScrollObserver,
|
|
nsISelectionListener, nsISupportsWeakReference);
|
|
|
|
// -----------------------------------------------------------------------------
|
|
// NoActionState
|
|
//
|
|
class AccessibleCaretEventHub::NoActionState
|
|
: public AccessibleCaretEventHub::State
|
|
{
|
|
public:
|
|
virtual const char* Name() const override { return "NoActionState"; }
|
|
|
|
virtual nsEventStatus OnPress(AccessibleCaretEventHub* aContext,
|
|
const nsPoint& aPoint,
|
|
int32_t aTouchId) override
|
|
{
|
|
nsEventStatus rv = nsEventStatus_eIgnore;
|
|
|
|
if (NS_SUCCEEDED(aContext->mManager->PressCaret(aPoint))) {
|
|
aContext->SetState(aContext->PressCaretState());
|
|
rv = nsEventStatus_eConsumeNoDefault;
|
|
} else {
|
|
aContext->SetState(aContext->PressNoCaretState());
|
|
}
|
|
|
|
aContext->mPressPoint = aPoint;
|
|
aContext->mActiveTouchId = aTouchId;
|
|
|
|
return rv;
|
|
}
|
|
|
|
virtual void OnScrollStart(AccessibleCaretEventHub* aContext) override
|
|
{
|
|
aContext->mManager->OnScrollStart();
|
|
aContext->SetState(aContext->ScrollState());
|
|
}
|
|
|
|
virtual void OnScrollPositionChanged(AccessibleCaretEventHub* aContext) override
|
|
{
|
|
aContext->mManager->OnScrollPositionChanged();
|
|
}
|
|
|
|
virtual void OnSelectionChanged(AccessibleCaretEventHub* aContext,
|
|
nsIDOMDocument* aDoc, nsISelection* aSel,
|
|
int16_t aReason) override
|
|
{
|
|
aContext->mManager->OnSelectionChanged(aDoc, aSel, aReason);
|
|
}
|
|
|
|
virtual void OnBlur(AccessibleCaretEventHub* aContext,
|
|
bool aIsLeavingDocument) override
|
|
{
|
|
aContext->mManager->OnBlur();
|
|
}
|
|
|
|
virtual void OnReflow(AccessibleCaretEventHub* aContext) override
|
|
{
|
|
aContext->mManager->OnReflow();
|
|
}
|
|
|
|
virtual void Enter(AccessibleCaretEventHub* aContext) override
|
|
{
|
|
aContext->mPressPoint = nsPoint(NS_UNCONSTRAINEDSIZE, NS_UNCONSTRAINEDSIZE);
|
|
aContext->mActiveTouchId = kInvalidTouchId;
|
|
}
|
|
};
|
|
|
|
// -----------------------------------------------------------------------------
|
|
// PressCaretState: Always consume the event since we've pressed on the caret.
|
|
//
|
|
class AccessibleCaretEventHub::PressCaretState
|
|
: public AccessibleCaretEventHub::State
|
|
{
|
|
public:
|
|
virtual const char* Name() const override { return "PressCaretState"; }
|
|
|
|
virtual nsEventStatus OnMove(AccessibleCaretEventHub* aContext,
|
|
const nsPoint& aPoint) override
|
|
{
|
|
if (aContext->MoveDistanceIsLarge(aPoint)) {
|
|
if (NS_SUCCEEDED(aContext->mManager->DragCaret(aPoint))) {
|
|
aContext->SetState(aContext->DragCaretState());
|
|
}
|
|
}
|
|
|
|
return nsEventStatus_eConsumeNoDefault;
|
|
}
|
|
|
|
virtual nsEventStatus OnRelease(AccessibleCaretEventHub* aContext) override
|
|
{
|
|
aContext->mManager->ReleaseCaret();
|
|
aContext->mManager->TapCaret(aContext->mPressPoint);
|
|
aContext->SetState(aContext->NoActionState());
|
|
|
|
return nsEventStatus_eConsumeNoDefault;
|
|
}
|
|
|
|
virtual nsEventStatus OnLongTap(AccessibleCaretEventHub* aContext,
|
|
const nsPoint& aPoint) override
|
|
{
|
|
return nsEventStatus_eConsumeNoDefault;
|
|
}
|
|
};
|
|
|
|
// -----------------------------------------------------------------------------
|
|
// DragCaretState: Always consume the event since we've pressed on the caret.
|
|
//
|
|
class AccessibleCaretEventHub::DragCaretState
|
|
: public AccessibleCaretEventHub::State
|
|
{
|
|
public:
|
|
virtual const char* Name() const override { return "DragCaretState"; }
|
|
|
|
virtual nsEventStatus OnMove(AccessibleCaretEventHub* aContext,
|
|
const nsPoint& aPoint) override
|
|
{
|
|
aContext->mManager->DragCaret(aPoint);
|
|
|
|
return nsEventStatus_eConsumeNoDefault;
|
|
}
|
|
|
|
virtual nsEventStatus OnRelease(AccessibleCaretEventHub* aContext) override
|
|
{
|
|
aContext->mManager->ReleaseCaret();
|
|
aContext->SetState(aContext->NoActionState());
|
|
|
|
return nsEventStatus_eConsumeNoDefault;
|
|
}
|
|
};
|
|
|
|
// -----------------------------------------------------------------------------
|
|
// PressNoCaretState
|
|
//
|
|
class AccessibleCaretEventHub::PressNoCaretState
|
|
: public AccessibleCaretEventHub::State
|
|
{
|
|
public:
|
|
virtual const char* Name() const override { return "PressNoCaretState"; }
|
|
|
|
virtual nsEventStatus OnMove(AccessibleCaretEventHub* aContext,
|
|
const nsPoint& aPoint) override
|
|
{
|
|
if (aContext->MoveDistanceIsLarge(aPoint)) {
|
|
aContext->SetState(aContext->NoActionState());
|
|
}
|
|
|
|
return nsEventStatus_eIgnore;
|
|
}
|
|
|
|
virtual nsEventStatus OnRelease(AccessibleCaretEventHub* aContext) override
|
|
{
|
|
aContext->SetState(aContext->NoActionState());
|
|
|
|
return nsEventStatus_eIgnore;
|
|
}
|
|
|
|
virtual nsEventStatus OnLongTap(AccessibleCaretEventHub* aContext,
|
|
const nsPoint& aPoint) override
|
|
{
|
|
aContext->SetState(aContext->LongTapState());
|
|
|
|
return aContext->GetState()->OnLongTap(aContext, aPoint);
|
|
}
|
|
|
|
virtual void OnScrollStart(AccessibleCaretEventHub* aContext) override
|
|
{
|
|
aContext->mManager->OnScrollStart();
|
|
aContext->SetState(aContext->ScrollState());
|
|
}
|
|
|
|
virtual void OnBlur(AccessibleCaretEventHub* aContext,
|
|
bool aIsLeavingDocument) override
|
|
{
|
|
aContext->mManager->OnBlur();
|
|
if (aIsLeavingDocument) {
|
|
aContext->SetState(aContext->NoActionState());
|
|
}
|
|
}
|
|
|
|
virtual void OnSelectionChanged(AccessibleCaretEventHub* aContext,
|
|
nsIDOMDocument* aDoc, nsISelection* aSel,
|
|
int16_t aReason) override
|
|
{
|
|
aContext->mManager->OnSelectionChanged(aDoc, aSel, aReason);
|
|
}
|
|
|
|
virtual void OnReflow(AccessibleCaretEventHub* aContext) override
|
|
{
|
|
aContext->mManager->OnReflow();
|
|
}
|
|
|
|
virtual void Enter(AccessibleCaretEventHub* aContext) override
|
|
{
|
|
aContext->LaunchLongTapInjector();
|
|
}
|
|
|
|
virtual void Leave(AccessibleCaretEventHub* aContext) override
|
|
{
|
|
aContext->CancelLongTapInjector();
|
|
}
|
|
};
|
|
|
|
// -----------------------------------------------------------------------------
|
|
// ScrollState
|
|
//
|
|
class AccessibleCaretEventHub::ScrollState
|
|
: public AccessibleCaretEventHub::State
|
|
{
|
|
public:
|
|
virtual const char* Name() const override { return "ScrollState"; }
|
|
|
|
virtual void OnScrollEnd(AccessibleCaretEventHub* aContext) override
|
|
{
|
|
aContext->SetState(aContext->PostScrollState());
|
|
}
|
|
|
|
virtual void OnBlur(AccessibleCaretEventHub* aContext,
|
|
bool aIsLeavingDocument) override
|
|
{
|
|
aContext->mManager->OnBlur();
|
|
if (aIsLeavingDocument) {
|
|
aContext->SetState(aContext->NoActionState());
|
|
}
|
|
}
|
|
};
|
|
|
|
// -----------------------------------------------------------------------------
|
|
// PostScrollState: In this state, we are waiting for another APZ start or press
|
|
// event.
|
|
//
|
|
class AccessibleCaretEventHub::PostScrollState
|
|
: public AccessibleCaretEventHub::State
|
|
{
|
|
public:
|
|
virtual const char* Name() const override { return "PostScrollState"; }
|
|
|
|
virtual nsEventStatus OnPress(AccessibleCaretEventHub* aContext,
|
|
const nsPoint& aPoint,
|
|
int32_t aTouchId) override
|
|
{
|
|
aContext->mManager->OnScrollEnd();
|
|
aContext->SetState(aContext->NoActionState());
|
|
|
|
return aContext->GetState()->OnPress(aContext, aPoint, aTouchId);
|
|
}
|
|
|
|
virtual void OnScrollStart(AccessibleCaretEventHub* aContext) override
|
|
{
|
|
aContext->SetState(aContext->ScrollState());
|
|
}
|
|
|
|
virtual void OnScrollEnd(AccessibleCaretEventHub* aContext) override
|
|
{
|
|
aContext->mManager->OnScrollEnd();
|
|
aContext->SetState(aContext->NoActionState());
|
|
}
|
|
|
|
virtual void OnBlur(AccessibleCaretEventHub* aContext,
|
|
bool aIsLeavingDocument) override
|
|
{
|
|
aContext->mManager->OnBlur();
|
|
if (aIsLeavingDocument) {
|
|
aContext->SetState(aContext->NoActionState());
|
|
}
|
|
}
|
|
|
|
virtual void Enter(AccessibleCaretEventHub* aContext) override
|
|
{
|
|
// Launch the injector to leave PostScrollState.
|
|
aContext->LaunchScrollEndInjector();
|
|
}
|
|
|
|
virtual void Leave(AccessibleCaretEventHub* aContext) override
|
|
{
|
|
aContext->CancelScrollEndInjector();
|
|
}
|
|
};
|
|
|
|
// -----------------------------------------------------------------------------
|
|
// LongTapState
|
|
//
|
|
class AccessibleCaretEventHub::LongTapState
|
|
: public AccessibleCaretEventHub::State
|
|
{
|
|
public:
|
|
virtual const char* Name() const override { return "LongTapState"; }
|
|
|
|
virtual nsEventStatus OnLongTap(AccessibleCaretEventHub* aContext,
|
|
const nsPoint& aPoint) override
|
|
{
|
|
nsEventStatus rv = nsEventStatus_eIgnore;
|
|
|
|
if (NS_SUCCEEDED(aContext->mManager->SelectWordOrShortcut(aPoint))) {
|
|
rv = nsEventStatus_eConsumeNoDefault;
|
|
}
|
|
|
|
aContext->SetState(aContext->NoActionState());
|
|
|
|
return rv;
|
|
}
|
|
|
|
virtual void OnReflow(AccessibleCaretEventHub* aContext) override
|
|
{
|
|
aContext->mManager->OnReflow();
|
|
}
|
|
};
|
|
|
|
// -----------------------------------------------------------------------------
|
|
// Implementation of AccessibleCaretEventHub methods
|
|
//
|
|
AccessibleCaretEventHub::State*
|
|
AccessibleCaretEventHub::GetState() const
|
|
{
|
|
return mState;
|
|
}
|
|
|
|
void
|
|
AccessibleCaretEventHub::SetState(State* aState)
|
|
{
|
|
MOZ_ASSERT(aState);
|
|
|
|
AC_LOG("%s -> %s", mState->Name(), aState->Name());
|
|
|
|
mState->Leave(this);
|
|
mState = aState;
|
|
mState->Enter(this);
|
|
}
|
|
|
|
MOZ_IMPL_STATE_CLASS_GETTER(NoActionState)
|
|
MOZ_IMPL_STATE_CLASS_GETTER(PressCaretState)
|
|
MOZ_IMPL_STATE_CLASS_GETTER(DragCaretState)
|
|
MOZ_IMPL_STATE_CLASS_GETTER(PressNoCaretState)
|
|
MOZ_IMPL_STATE_CLASS_GETTER(ScrollState)
|
|
MOZ_IMPL_STATE_CLASS_GETTER(PostScrollState)
|
|
MOZ_IMPL_STATE_CLASS_GETTER(LongTapState)
|
|
|
|
bool AccessibleCaretEventHub::sUseLongTapInjector = true;
|
|
|
|
AccessibleCaretEventHub::AccessibleCaretEventHub(nsIPresShell* aPresShell)
|
|
: mPresShell(aPresShell)
|
|
{
|
|
static bool prefsAdded = false;
|
|
if (!prefsAdded) {
|
|
Preferences::AddBoolVarCache(&sUseLongTapInjector,
|
|
"layout.accessiblecaret.use_long_tap_injector");
|
|
prefsAdded = true;
|
|
}
|
|
}
|
|
|
|
AccessibleCaretEventHub::~AccessibleCaretEventHub()
|
|
{
|
|
}
|
|
|
|
void
|
|
AccessibleCaretEventHub::Init()
|
|
{
|
|
if (mInitialized || !mPresShell || !mPresShell->GetCanvasFrame() ||
|
|
!mPresShell->GetCanvasFrame()->GetCustomContentContainer()) {
|
|
return;
|
|
}
|
|
|
|
// Without nsAutoScriptBlocker, the script might be run after constructing
|
|
// mFirstCaret in AccessibleCaretManager's constructor, which might destructs
|
|
// the whole frame tree. Therefore we'll fail to construct mSecondCaret
|
|
// because we cannot get root frame or canvas frame from mPresShell to inject
|
|
// anonymous content. To avoid that, we protect Init() by nsAutoScriptBlocker.
|
|
// To reproduce, run "./mach crashtest layout/base/crashtests/897852.html"
|
|
// without the following scriptBlocker.
|
|
nsAutoScriptBlocker scriptBlocker;
|
|
|
|
nsPresContext* presContext = mPresShell->GetPresContext();
|
|
MOZ_ASSERT(presContext, "PresContext should be given in PresShell::Init()");
|
|
|
|
nsIDocShell* docShell = presContext->GetDocShell();
|
|
if (!docShell) {
|
|
return;
|
|
}
|
|
|
|
docShell->AddWeakReflowObserver(this);
|
|
docShell->AddWeakScrollObserver(this);
|
|
|
|
mDocShell = static_cast<nsDocShell*>(docShell);
|
|
|
|
if (sUseLongTapInjector) {
|
|
mLongTapInjectorTimer = do_CreateInstance("@mozilla.org/timer;1");
|
|
}
|
|
|
|
mScrollEndInjectorTimer = do_CreateInstance("@mozilla.org/timer;1");
|
|
|
|
mManager = MakeUnique<AccessibleCaretManager>(mPresShell);
|
|
|
|
mInitialized = true;
|
|
}
|
|
|
|
void
|
|
AccessibleCaretEventHub::Terminate()
|
|
{
|
|
if (!mInitialized) {
|
|
return;
|
|
}
|
|
|
|
RefPtr<nsDocShell> docShell(mDocShell.get());
|
|
if (docShell) {
|
|
docShell->RemoveWeakReflowObserver(this);
|
|
docShell->RemoveWeakScrollObserver(this);
|
|
}
|
|
|
|
if (mLongTapInjectorTimer) {
|
|
mLongTapInjectorTimer->Cancel();
|
|
}
|
|
|
|
if (mScrollEndInjectorTimer) {
|
|
mScrollEndInjectorTimer->Cancel();
|
|
}
|
|
|
|
mManager = nullptr;
|
|
mPresShell = nullptr;
|
|
mInitialized = false;
|
|
}
|
|
|
|
nsEventStatus
|
|
AccessibleCaretEventHub::HandleEvent(WidgetEvent* aEvent)
|
|
{
|
|
nsEventStatus status = nsEventStatus_eIgnore;
|
|
|
|
if (!mInitialized) {
|
|
return status;
|
|
}
|
|
|
|
switch (aEvent->mClass) {
|
|
case eMouseEventClass:
|
|
status = HandleMouseEvent(aEvent->AsMouseEvent());
|
|
break;
|
|
|
|
case eTouchEventClass:
|
|
status = HandleTouchEvent(aEvent->AsTouchEvent());
|
|
break;
|
|
|
|
case eKeyboardEventClass:
|
|
status = HandleKeyboardEvent(aEvent->AsKeyboardEvent());
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
|
|
return status;
|
|
}
|
|
|
|
nsEventStatus
|
|
AccessibleCaretEventHub::HandleMouseEvent(WidgetMouseEvent* aEvent)
|
|
{
|
|
nsEventStatus rv = nsEventStatus_eIgnore;
|
|
|
|
if (aEvent->button != WidgetMouseEvent::eLeftButton) {
|
|
return rv;
|
|
}
|
|
|
|
int32_t id = (mActiveTouchId == kInvalidTouchId ?
|
|
kDefaultTouchId : mActiveTouchId);
|
|
nsPoint point = GetMouseEventPosition(aEvent);
|
|
|
|
switch (aEvent->mMessage) {
|
|
case eMouseDown:
|
|
AC_LOGV("Before eMouseDown, state: %s", mState->Name());
|
|
rv = mState->OnPress(this, point, id);
|
|
AC_LOGV("After eMouseDown, state: %s, consume: %d",
|
|
mState->Name(), rv);
|
|
break;
|
|
|
|
case eMouseMove:
|
|
AC_LOGV("Before eMouseMove, state: %s", mState->Name());
|
|
rv = mState->OnMove(this, point);
|
|
AC_LOGV("After eMouseMove, state: %s, consume: %d", mState->Name(), rv);
|
|
break;
|
|
|
|
case eMouseUp:
|
|
AC_LOGV("Before eMouseUp, state: %s", mState->Name());
|
|
rv = mState->OnRelease(this);
|
|
AC_LOGV("After eMouseUp, state: %s, consume: %d", mState->Name(),
|
|
rv);
|
|
break;
|
|
|
|
case eMouseLongTap:
|
|
AC_LOGV("Before eMouseLongTap, state: %s", mState->Name());
|
|
rv = mState->OnLongTap(this, point);
|
|
AC_LOGV("After eMouseLongTap, state: %s, consume: %d", mState->Name(),
|
|
rv);
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
|
|
return rv;
|
|
}
|
|
|
|
nsEventStatus
|
|
AccessibleCaretEventHub::HandleTouchEvent(WidgetTouchEvent* aEvent)
|
|
{
|
|
nsEventStatus rv = nsEventStatus_eIgnore;
|
|
|
|
int32_t id = (mActiveTouchId == kInvalidTouchId ?
|
|
aEvent->touches[0]->Identifier() : mActiveTouchId);
|
|
nsPoint point = GetTouchEventPosition(aEvent, id);
|
|
|
|
switch (aEvent->mMessage) {
|
|
case eTouchStart:
|
|
AC_LOGV("Before eTouchStart, state: %s", mState->Name());
|
|
rv = mState->OnPress(this, point, id);
|
|
AC_LOGV("After eTouchStart, state: %s, consume: %d", mState->Name(), rv);
|
|
break;
|
|
|
|
case eTouchMove:
|
|
AC_LOGV("Before eTouchMove, state: %s", mState->Name());
|
|
rv = mState->OnMove(this, point);
|
|
AC_LOGV("After eTouchMove, state: %s, consume: %d", mState->Name(), rv);
|
|
break;
|
|
|
|
case eTouchEnd:
|
|
AC_LOGV("Before eTouchEnd, state: %s", mState->Name());
|
|
rv = mState->OnRelease(this);
|
|
AC_LOGV("After eTouchEnd, state: %s, consume: %d", mState->Name(), rv);
|
|
break;
|
|
|
|
case eTouchCancel:
|
|
AC_LOGV("Before eTouchCancel, state: %s", mState->Name());
|
|
rv = mState->OnRelease(this);
|
|
AC_LOGV("After eTouchCancel, state: %s, consume: %d", mState->Name(),
|
|
rv);
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
|
|
return rv;
|
|
}
|
|
|
|
nsEventStatus
|
|
AccessibleCaretEventHub::HandleKeyboardEvent(WidgetKeyboardEvent* aEvent)
|
|
{
|
|
switch (aEvent->mMessage) {
|
|
case eKeyUp:
|
|
AC_LOGV("eKeyUp, state: %s", mState->Name());
|
|
mManager->OnKeyboardEvent();
|
|
break;
|
|
|
|
case eKeyDown:
|
|
AC_LOGV("eKeyDown, state: %s", mState->Name());
|
|
mManager->OnKeyboardEvent();
|
|
break;
|
|
|
|
case eKeyPress:
|
|
AC_LOGV("eKeyPress, state: %s", mState->Name());
|
|
mManager->OnKeyboardEvent();
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
|
|
return nsEventStatus_eIgnore;
|
|
}
|
|
|
|
bool
|
|
AccessibleCaretEventHub::MoveDistanceIsLarge(const nsPoint& aPoint) const
|
|
{
|
|
nsPoint delta = aPoint - mPressPoint;
|
|
return NS_hypot(delta.x, delta.y) >
|
|
nsPresContext::AppUnitsPerCSSPixel() * kMoveStartToleranceInPixel;
|
|
}
|
|
|
|
void
|
|
AccessibleCaretEventHub::LaunchLongTapInjector()
|
|
{
|
|
if (!mLongTapInjectorTimer) {
|
|
return;
|
|
}
|
|
|
|
int32_t longTapDelay = gfxPrefs::UiClickHoldContextMenusDelay();
|
|
mLongTapInjectorTimer->InitWithFuncCallback(FireLongTap, this, longTapDelay,
|
|
nsITimer::TYPE_ONE_SHOT);
|
|
}
|
|
|
|
void
|
|
AccessibleCaretEventHub::CancelLongTapInjector()
|
|
{
|
|
if (!mLongTapInjectorTimer) {
|
|
return;
|
|
}
|
|
|
|
mLongTapInjectorTimer->Cancel();
|
|
}
|
|
|
|
/* static */ void
|
|
AccessibleCaretEventHub::FireLongTap(nsITimer* aTimer,
|
|
void* aAccessibleCaretEventHub)
|
|
{
|
|
auto self = static_cast<AccessibleCaretEventHub*>(aAccessibleCaretEventHub);
|
|
self->mState->OnLongTap(self, self->mPressPoint);
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
AccessibleCaretEventHub::Reflow(DOMHighResTimeStamp aStart,
|
|
DOMHighResTimeStamp aEnd)
|
|
{
|
|
if (!mInitialized) {
|
|
return NS_OK;
|
|
}
|
|
|
|
AC_LOG("%s, state: %s", __FUNCTION__, mState->Name());
|
|
mState->OnReflow(this);
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
AccessibleCaretEventHub::ReflowInterruptible(DOMHighResTimeStamp aStart,
|
|
DOMHighResTimeStamp aEnd)
|
|
{
|
|
if (!mInitialized) {
|
|
return NS_OK;
|
|
}
|
|
|
|
return Reflow(aStart, aEnd);
|
|
}
|
|
|
|
void
|
|
AccessibleCaretEventHub::AsyncPanZoomStarted()
|
|
{
|
|
if (!mInitialized) {
|
|
return;
|
|
}
|
|
|
|
AC_LOG("%s, state: %s", __FUNCTION__, mState->Name());
|
|
mState->OnScrollStart(this);
|
|
}
|
|
|
|
void
|
|
AccessibleCaretEventHub::AsyncPanZoomStopped()
|
|
{
|
|
if (!mInitialized) {
|
|
return;
|
|
}
|
|
|
|
AC_LOG("%s, state: %s", __FUNCTION__, mState->Name());
|
|
mState->OnScrollEnd(this);
|
|
}
|
|
|
|
void
|
|
AccessibleCaretEventHub::ScrollPositionChanged()
|
|
{
|
|
if (!mInitialized) {
|
|
return;
|
|
}
|
|
|
|
AC_LOG("%s, state: %s", __FUNCTION__, mState->Name());
|
|
mState->OnScrollPositionChanged(this);
|
|
}
|
|
|
|
void
|
|
AccessibleCaretEventHub::LaunchScrollEndInjector()
|
|
{
|
|
if (!mScrollEndInjectorTimer) {
|
|
return;
|
|
}
|
|
|
|
mScrollEndInjectorTimer->InitWithFuncCallback(
|
|
FireScrollEnd, this, kScrollEndTimerDelay, nsITimer::TYPE_ONE_SHOT);
|
|
}
|
|
|
|
void
|
|
AccessibleCaretEventHub::CancelScrollEndInjector()
|
|
{
|
|
if (!mScrollEndInjectorTimer) {
|
|
return;
|
|
}
|
|
|
|
mScrollEndInjectorTimer->Cancel();
|
|
}
|
|
|
|
/* static */ void
|
|
AccessibleCaretEventHub::FireScrollEnd(nsITimer* aTimer,
|
|
void* aAccessibleCaretEventHub)
|
|
{
|
|
auto self = static_cast<AccessibleCaretEventHub*>(aAccessibleCaretEventHub);
|
|
self->mState->OnScrollEnd(self);
|
|
}
|
|
|
|
nsresult
|
|
AccessibleCaretEventHub::NotifySelectionChanged(nsIDOMDocument* aDoc,
|
|
nsISelection* aSel,
|
|
int16_t aReason)
|
|
{
|
|
if (!mInitialized) {
|
|
return NS_OK;
|
|
}
|
|
|
|
AC_LOG("%s, state: %s, reason: %d", __FUNCTION__, mState->Name(), aReason);
|
|
mState->OnSelectionChanged(this, aDoc, aSel, aReason);
|
|
return NS_OK;
|
|
}
|
|
|
|
void
|
|
AccessibleCaretEventHub::NotifyBlur(bool aIsLeavingDocument)
|
|
{
|
|
if (!mInitialized) {
|
|
return;
|
|
}
|
|
|
|
AC_LOG("%s, state: %s", __FUNCTION__, mState->Name());
|
|
mState->OnBlur(this, aIsLeavingDocument);
|
|
}
|
|
|
|
nsPoint
|
|
AccessibleCaretEventHub::GetTouchEventPosition(WidgetTouchEvent* aEvent,
|
|
int32_t aIdentifier) const
|
|
{
|
|
for (dom::Touch* touch : aEvent->touches) {
|
|
if (touch->Identifier() == aIdentifier) {
|
|
LayoutDeviceIntPoint touchIntPoint = touch->mRefPoint;
|
|
|
|
// Get event coordinate relative to root frame.
|
|
nsIFrame* rootFrame = mPresShell->GetRootFrame();
|
|
return nsLayoutUtils::GetEventCoordinatesRelativeTo(aEvent, touchIntPoint,
|
|
rootFrame);
|
|
}
|
|
}
|
|
return nsPoint(NS_UNCONSTRAINEDSIZE, NS_UNCONSTRAINEDSIZE);
|
|
}
|
|
|
|
nsPoint
|
|
AccessibleCaretEventHub::GetMouseEventPosition(WidgetMouseEvent* aEvent) const
|
|
{
|
|
LayoutDeviceIntPoint mouseIntPoint = aEvent->AsGUIEvent()->refPoint;
|
|
|
|
// Get event coordinate relative to root frame.
|
|
nsIFrame* rootFrame = mPresShell->GetRootFrame();
|
|
return nsLayoutUtils::GetEventCoordinatesRelativeTo(aEvent, mouseIntPoint,
|
|
rootFrame);
|
|
}
|
|
|
|
} // namespace mozilla
|