mirror of
https://github.com/mozilla/gecko-dev.git
synced 2025-04-03 21:22:47 +00:00
Bug 571294 - Part 1: Implement selection events behind the dom.select_events.enabled pref, r=smaug
This commit is contained in:
parent
7b2f24f27d
commit
fe31896607
@ -814,6 +814,7 @@ nsIContent::PreHandleEvent(EventChainPreVisitor& aVisitor)
|
|||||||
case eFormReset:
|
case eFormReset:
|
||||||
case eResize:
|
case eResize:
|
||||||
case eScroll:
|
case eScroll:
|
||||||
|
case NS_SELECT_START:
|
||||||
stopEvent = true;
|
stopEvent = true;
|
||||||
break;
|
break;
|
||||||
case eUnidentifiedEvent:
|
case eUnidentifiedEvent:
|
||||||
@ -827,8 +828,7 @@ nsIContent::PreHandleEvent(EventChainPreVisitor& aVisitor)
|
|||||||
eventType.EqualsLiteral("load") ||
|
eventType.EqualsLiteral("load") ||
|
||||||
eventType.EqualsLiteral("reset") ||
|
eventType.EqualsLiteral("reset") ||
|
||||||
eventType.EqualsLiteral("resize") ||
|
eventType.EqualsLiteral("resize") ||
|
||||||
eventType.EqualsLiteral("scroll") ||
|
eventType.EqualsLiteral("scroll")) {
|
||||||
eventType.EqualsLiteral("selectstart")) {
|
|
||||||
stopEvent = true;
|
stopEvent = true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -882,6 +882,8 @@ GK_ATOM(onscanningstatechanged, "onscanningstatechanged")
|
|||||||
GK_ATOM(onscostatuschanged, "onscostatuschanged")
|
GK_ATOM(onscostatuschanged, "onscostatuschanged")
|
||||||
GK_ATOM(onscroll, "onscroll")
|
GK_ATOM(onscroll, "onscroll")
|
||||||
GK_ATOM(onselect, "onselect")
|
GK_ATOM(onselect, "onselect")
|
||||||
|
GK_ATOM(onselectionchange, "onselectionchange")
|
||||||
|
GK_ATOM(onselectstart, "onselectstart")
|
||||||
GK_ATOM(onsending, "onsending")
|
GK_ATOM(onsending, "onsending")
|
||||||
GK_ATOM(onsent, "onsent")
|
GK_ATOM(onsent, "onsent")
|
||||||
GK_ATOM(onset, "onset")
|
GK_ATOM(onset, "onset")
|
||||||
|
@ -31,6 +31,7 @@
|
|||||||
#include "mozilla/dom/RangeBinding.h"
|
#include "mozilla/dom/RangeBinding.h"
|
||||||
#include "mozilla/dom/DOMRect.h"
|
#include "mozilla/dom/DOMRect.h"
|
||||||
#include "mozilla/dom/ShadowRoot.h"
|
#include "mozilla/dom/ShadowRoot.h"
|
||||||
|
#include "mozilla/dom/Selection.h"
|
||||||
#include "mozilla/Telemetry.h"
|
#include "mozilla/Telemetry.h"
|
||||||
#include "mozilla/Likely.h"
|
#include "mozilla/Likely.h"
|
||||||
#include "nsCSSFrameConstructor.h"
|
#include "nsCSSFrameConstructor.h"
|
||||||
@ -194,6 +195,26 @@ nsRange::~nsRange()
|
|||||||
DoSetRange(nullptr, 0, nullptr, 0, nullptr);
|
DoSetRange(nullptr, 0, nullptr, 0, nullptr);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
nsRange::nsRange(nsINode* aNode)
|
||||||
|
: mRoot(nullptr)
|
||||||
|
, mStartOffset(0)
|
||||||
|
, mEndOffset(0)
|
||||||
|
, mIsPositioned(false)
|
||||||
|
, mIsDetached(false)
|
||||||
|
, mMaySpanAnonymousSubtrees(false)
|
||||||
|
, mIsGenerated(false)
|
||||||
|
, mStartOffsetWasIncremented(false)
|
||||||
|
, mEndOffsetWasIncremented(false)
|
||||||
|
, mEnableGravitationOnElementRemoval(true)
|
||||||
|
#ifdef DEBUG
|
||||||
|
, mAssertNextInsertOrAppendIndex(-1)
|
||||||
|
, mAssertNextInsertOrAppendNode(nullptr)
|
||||||
|
#endif
|
||||||
|
{
|
||||||
|
MOZ_ASSERT(aNode, "range isn't in a document!");
|
||||||
|
mOwner = aNode->OwnerDoc();
|
||||||
|
}
|
||||||
|
|
||||||
/* static */
|
/* static */
|
||||||
nsresult
|
nsresult
|
||||||
nsRange::CreateRange(nsINode* aStartParent, int32_t aStartOffset,
|
nsRange::CreateRange(nsINode* aStartParent, int32_t aStartOffset,
|
||||||
@ -269,6 +290,10 @@ NS_IMPL_CYCLE_COLLECTION_UNLINK_BEGIN(nsRange)
|
|||||||
NS_IMPL_CYCLE_COLLECTION_UNLINK_PRESERVED_WRAPPER
|
NS_IMPL_CYCLE_COLLECTION_UNLINK_PRESERVED_WRAPPER
|
||||||
NS_IMPL_CYCLE_COLLECTION_UNLINK(mOwner);
|
NS_IMPL_CYCLE_COLLECTION_UNLINK(mOwner);
|
||||||
tmp->Reset();
|
tmp->Reset();
|
||||||
|
|
||||||
|
// This needs to be unlinked after Reset() is called, as it controls
|
||||||
|
// the result of IsInSelection() which is used by tmp->Reset().
|
||||||
|
NS_IMPL_CYCLE_COLLECTION_UNLINK(mSelection);
|
||||||
NS_IMPL_CYCLE_COLLECTION_UNLINK_END
|
NS_IMPL_CYCLE_COLLECTION_UNLINK_END
|
||||||
|
|
||||||
NS_IMPL_CYCLE_COLLECTION_TRAVERSE_BEGIN(nsRange)
|
NS_IMPL_CYCLE_COLLECTION_TRAVERSE_BEGIN(nsRange)
|
||||||
@ -276,6 +301,7 @@ NS_IMPL_CYCLE_COLLECTION_TRAVERSE_BEGIN(nsRange)
|
|||||||
NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mStartParent)
|
NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mStartParent)
|
||||||
NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mEndParent)
|
NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mEndParent)
|
||||||
NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mRoot)
|
NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mRoot)
|
||||||
|
NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mSelection)
|
||||||
NS_IMPL_CYCLE_COLLECTION_TRAVERSE_SCRIPT_OBJECTS
|
NS_IMPL_CYCLE_COLLECTION_TRAVERSE_SCRIPT_OBJECTS
|
||||||
NS_IMPL_CYCLE_COLLECTION_TRAVERSE_END
|
NS_IMPL_CYCLE_COLLECTION_TRAVERSE_END
|
||||||
|
|
||||||
@ -879,14 +905,20 @@ nsRange::DoSetRange(nsINode* aStartN, int32_t aStartOffset,
|
|||||||
RegisterCommonAncestor(newCommonAncestor);
|
RegisterCommonAncestor(newCommonAncestor);
|
||||||
} else {
|
} else {
|
||||||
NS_ASSERTION(!mIsPositioned, "unexpected disconnected nodes");
|
NS_ASSERTION(!mIsPositioned, "unexpected disconnected nodes");
|
||||||
mInSelection = false;
|
mSelection = nullptr;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// This needs to be the last thing this function does. See comment
|
// This needs to be the last thing this function does, other than notifying
|
||||||
// in ParentChainChanged.
|
// selection listeners. See comment in ParentChainChanged.
|
||||||
mRoot = aRoot;
|
mRoot = aRoot;
|
||||||
|
|
||||||
|
// Notify any selection listeners. This has to occur last because otherwise the world
|
||||||
|
// could be observed by a selection listener while the range was in an invalid state.
|
||||||
|
if (mSelection) {
|
||||||
|
mSelection->NotifySelectionListeners();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static int32_t
|
static int32_t
|
||||||
@ -897,6 +929,28 @@ IndexOf(nsINode* aChild)
|
|||||||
return parent ? parent->IndexOf(aChild) : -1;
|
return parent ? parent->IndexOf(aChild) : -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
nsRange::SetSelection(mozilla::dom::Selection* aSelection)
|
||||||
|
{
|
||||||
|
if (mSelection == aSelection) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
// At least one of aSelection and mSelection must be null
|
||||||
|
// aSelection will be null when we are removing from a selection
|
||||||
|
// and a range can't be in more than one selection at a time,
|
||||||
|
// thus mSelection must be null too.
|
||||||
|
MOZ_ASSERT(!aSelection || !mSelection);
|
||||||
|
|
||||||
|
mSelection = aSelection;
|
||||||
|
nsINode* commonAncestor = GetCommonAncestor();
|
||||||
|
NS_ASSERTION(commonAncestor, "unexpected disconnected nodes");
|
||||||
|
if (mSelection) {
|
||||||
|
RegisterCommonAncestor(commonAncestor);
|
||||||
|
} else {
|
||||||
|
UnregisterCommonAncestor(commonAncestor);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
nsINode*
|
nsINode*
|
||||||
nsRange::GetCommonAncestor() const
|
nsRange::GetCommonAncestor() const
|
||||||
{
|
{
|
||||||
|
@ -28,6 +28,7 @@ namespace dom {
|
|||||||
class DocumentFragment;
|
class DocumentFragment;
|
||||||
class DOMRect;
|
class DOMRect;
|
||||||
class DOMRectList;
|
class DOMRectList;
|
||||||
|
class Selection;
|
||||||
} // namespace dom
|
} // namespace dom
|
||||||
} // namespace mozilla
|
} // namespace mozilla
|
||||||
|
|
||||||
@ -42,26 +43,7 @@ class nsRange final : public nsIDOMRange,
|
|||||||
virtual ~nsRange();
|
virtual ~nsRange();
|
||||||
|
|
||||||
public:
|
public:
|
||||||
explicit nsRange(nsINode* aNode)
|
explicit nsRange(nsINode* aNode);
|
||||||
: mRoot(nullptr)
|
|
||||||
, mStartOffset(0)
|
|
||||||
, mEndOffset(0)
|
|
||||||
, mIsPositioned(false)
|
|
||||||
, mIsDetached(false)
|
|
||||||
, mMaySpanAnonymousSubtrees(false)
|
|
||||||
, mInSelection(false)
|
|
||||||
, mIsGenerated(false)
|
|
||||||
, mStartOffsetWasIncremented(false)
|
|
||||||
, mEndOffsetWasIncremented(false)
|
|
||||||
, mEnableGravitationOnElementRemoval(true)
|
|
||||||
#ifdef DEBUG
|
|
||||||
, mAssertNextInsertOrAppendIndex(-1)
|
|
||||||
, mAssertNextInsertOrAppendNode(nullptr)
|
|
||||||
#endif
|
|
||||||
{
|
|
||||||
MOZ_ASSERT(aNode, "range isn't in a document!");
|
|
||||||
mOwner = aNode->OwnerDoc();
|
|
||||||
}
|
|
||||||
|
|
||||||
static nsresult CreateRange(nsIDOMNode* aStartParent, int32_t aStartOffset,
|
static nsresult CreateRange(nsIDOMNode* aStartParent, int32_t aStartOffset,
|
||||||
nsIDOMNode* aEndParent, int32_t aEndOffset,
|
nsIDOMNode* aEndParent, int32_t aEndOffset,
|
||||||
@ -129,31 +111,18 @@ public:
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Return true iff this range is part of at least one Selection object
|
* Return true iff this range is part of a Selection object
|
||||||
* and isn't detached.
|
* and isn't detached.
|
||||||
*/
|
*/
|
||||||
bool IsInSelection() const
|
bool IsInSelection() const
|
||||||
{
|
{
|
||||||
return mInSelection;
|
return !!mSelection;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Called when the range is added/removed from a Selection.
|
* Called when the range is added/removed from a Selection.
|
||||||
*/
|
*/
|
||||||
void SetInSelection(bool aInSelection)
|
void SetSelection(mozilla::dom::Selection* aSelection);
|
||||||
{
|
|
||||||
if (mInSelection == aInSelection) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
mInSelection = aInSelection;
|
|
||||||
nsINode* commonAncestor = GetCommonAncestor();
|
|
||||||
NS_ASSERTION(commonAncestor, "unexpected disconnected nodes");
|
|
||||||
if (mInSelection) {
|
|
||||||
RegisterCommonAncestor(commonAncestor);
|
|
||||||
} else {
|
|
||||||
UnregisterCommonAncestor(commonAncestor);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Return true if this range was generated.
|
* Return true if this range was generated.
|
||||||
@ -349,13 +318,13 @@ protected:
|
|||||||
nsCOMPtr<nsINode> mRoot;
|
nsCOMPtr<nsINode> mRoot;
|
||||||
nsCOMPtr<nsINode> mStartParent;
|
nsCOMPtr<nsINode> mStartParent;
|
||||||
nsCOMPtr<nsINode> mEndParent;
|
nsCOMPtr<nsINode> mEndParent;
|
||||||
|
nsRefPtr<mozilla::dom::Selection> mSelection;
|
||||||
int32_t mStartOffset;
|
int32_t mStartOffset;
|
||||||
int32_t mEndOffset;
|
int32_t mEndOffset;
|
||||||
|
|
||||||
bool mIsPositioned : 1;
|
bool mIsPositioned : 1;
|
||||||
bool mIsDetached : 1;
|
bool mIsDetached : 1;
|
||||||
bool mMaySpanAnonymousSubtrees : 1;
|
bool mMaySpanAnonymousSubtrees : 1;
|
||||||
bool mInSelection : 1;
|
|
||||||
bool mIsGenerated : 1;
|
bool mIsGenerated : 1;
|
||||||
bool mStartOffsetWasIncremented : 1;
|
bool mStartOffsetWasIncremented : 1;
|
||||||
bool mEndOffsetWasIncremented : 1;
|
bool mEndOffsetWasIncremented : 1;
|
||||||
|
@ -350,6 +350,10 @@ EVENT(lostpointercapture,
|
|||||||
ePointerLostCapture,
|
ePointerLostCapture,
|
||||||
EventNameType_All,
|
EventNameType_All,
|
||||||
ePointerEventClass)
|
ePointerEventClass)
|
||||||
|
EVENT(selectstart,
|
||||||
|
NS_SELECT_START,
|
||||||
|
EventNameType_HTMLXUL,
|
||||||
|
eBasicEventClass)
|
||||||
|
|
||||||
// Not supported yet; probably never because "wheel" is a better idea.
|
// Not supported yet; probably never because "wheel" is a better idea.
|
||||||
// EVENT(mousewheel)
|
// EVENT(mousewheel)
|
||||||
@ -584,6 +588,10 @@ DOCUMENT_ONLY_EVENT(readystatechange,
|
|||||||
eReadyStateChange,
|
eReadyStateChange,
|
||||||
EventNameType_HTMLXUL,
|
EventNameType_HTMLXUL,
|
||||||
eBasicEventClass)
|
eBasicEventClass)
|
||||||
|
DOCUMENT_ONLY_EVENT(selectionchange,
|
||||||
|
NS_SELECTION_CHANGE,
|
||||||
|
EventNameType_HTMLXUL,
|
||||||
|
eBasicEventClass)
|
||||||
|
|
||||||
NON_IDL_EVENT(MozMouseHittest,
|
NON_IDL_EVENT(MozMouseHittest,
|
||||||
eMouseHitTest,
|
eMouseHitTest,
|
||||||
|
@ -151,6 +151,9 @@ partial interface Document {
|
|||||||
attribute EventHandler onpaste;
|
attribute EventHandler onpaste;
|
||||||
attribute EventHandler onbeforescriptexecute;
|
attribute EventHandler onbeforescriptexecute;
|
||||||
attribute EventHandler onafterscriptexecute;
|
attribute EventHandler onafterscriptexecute;
|
||||||
|
|
||||||
|
[Pref="dom.select_events.enabled"]
|
||||||
|
attribute EventHandler onselectionchange;
|
||||||
/**
|
/**
|
||||||
* True if this document is synthetic : stand alone image, video, audio file,
|
* True if this document is synthetic : stand alone image, video, audio file,
|
||||||
* etc.
|
* etc.
|
||||||
|
@ -89,6 +89,9 @@ interface GlobalEventHandlers {
|
|||||||
attribute EventHandler onvolumechange;
|
attribute EventHandler onvolumechange;
|
||||||
attribute EventHandler onwaiting;
|
attribute EventHandler onwaiting;
|
||||||
|
|
||||||
|
[Pref="dom.select_events.enabled"]
|
||||||
|
attribute EventHandler onselectstart;
|
||||||
|
|
||||||
// Pointer events handlers
|
// Pointer events handlers
|
||||||
[Pref="dom.w3c_pointer_events.enabled"]
|
[Pref="dom.w3c_pointer_events.enabled"]
|
||||||
attribute EventHandler onpointercancel;
|
attribute EventHandler onpointercancel;
|
||||||
|
@ -831,6 +831,14 @@ nsTextEditRules::WillDeleteSelection(Selection* aSelection,
|
|||||||
}
|
}
|
||||||
|
|
||||||
nsresult res = NS_OK;
|
nsresult res = NS_OK;
|
||||||
|
// If the current selection is empty (e.g the user presses backspace with
|
||||||
|
// a collapsed selection), then we want to avoid sending the selectstart
|
||||||
|
// event to the user, so we hide selection changes. However, we still
|
||||||
|
// want to send a single selectionchange event to the document, so we
|
||||||
|
// batch the selectionchange events, such that a single event fires after
|
||||||
|
// the AutoHideSelectionChanges destructor has been run.
|
||||||
|
SelectionBatcher selectionBatcher(aSelection);
|
||||||
|
AutoHideSelectionChanges hideSelection(aSelection);
|
||||||
nsAutoScriptBlocker scriptBlocker;
|
nsAutoScriptBlocker scriptBlocker;
|
||||||
|
|
||||||
if (IsPasswordEditor())
|
if (IsPasswordEditor())
|
||||||
|
@ -946,13 +946,20 @@ mozInlineSpellChecker::ReplaceWord(nsIDOMNode *aNode, int32_t aOffset,
|
|||||||
|
|
||||||
if (range)
|
if (range)
|
||||||
{
|
{
|
||||||
|
// This range was retrieved from the spellchecker selection. As
|
||||||
|
// ranges cannot be shared between selections, we must clone it
|
||||||
|
// before adding it to the editor's selection.
|
||||||
|
nsCOMPtr<nsIDOMRange> editorRange;
|
||||||
|
res = range->CloneRange(getter_AddRefs(editorRange));
|
||||||
|
NS_ENSURE_SUCCESS(res, res);
|
||||||
|
|
||||||
nsAutoPlaceHolderBatch phb(editor, nullptr);
|
nsAutoPlaceHolderBatch phb(editor, nullptr);
|
||||||
|
|
||||||
nsCOMPtr<nsISelection> selection;
|
nsCOMPtr<nsISelection> selection;
|
||||||
res = editor->GetSelection(getter_AddRefs(selection));
|
res = editor->GetSelection(getter_AddRefs(selection));
|
||||||
NS_ENSURE_SUCCESS(res, res);
|
NS_ENSURE_SUCCESS(res, res);
|
||||||
selection->RemoveAllRanges();
|
selection->RemoveAllRanges();
|
||||||
selection->AddRange(range);
|
selection->AddRange(editorRange);
|
||||||
editor->DeleteSelection(nsIEditor::eNone, nsIEditor::eStrip);
|
editor->DeleteSelection(nsIEditor::eNone, nsIEditor::eStrip);
|
||||||
|
|
||||||
nsCOMPtr<nsIPlaintextEditor> textEditor(do_QueryReferent(mEditor));
|
nsCOMPtr<nsIPlaintextEditor> textEditor(do_QueryReferent(mEditor));
|
||||||
|
@ -108,7 +108,7 @@ public:
|
|||||||
* into multiple ranges to exclude those before adding the resulting ranges
|
* into multiple ranges to exclude those before adding the resulting ranges
|
||||||
* to this Selection.
|
* to this Selection.
|
||||||
*/
|
*/
|
||||||
nsresult AddItem(nsRange* aRange, int32_t* aOutIndex);
|
nsresult AddItem(nsRange* aRange, int32_t* aOutIndex, bool aNoStartSelect = false);
|
||||||
nsresult RemoveItem(nsRange* aRange);
|
nsresult RemoveItem(nsRange* aRange);
|
||||||
nsresult RemoveCollapsedRanges();
|
nsresult RemoveCollapsedRanges();
|
||||||
nsresult Clear(nsPresContext* aPresContext);
|
nsresult Clear(nsPresContext* aPresContext);
|
||||||
@ -204,6 +204,9 @@ public:
|
|||||||
int16_t aVPercent, int16_t aHPercent,
|
int16_t aVPercent, int16_t aHPercent,
|
||||||
mozilla::ErrorResult& aRv);
|
mozilla::ErrorResult& aRv);
|
||||||
|
|
||||||
|
void AddSelectionChangeBlocker();
|
||||||
|
void RemoveSelectionChangeBlocker();
|
||||||
|
bool IsBlockingSelectionChangeEvents() const;
|
||||||
private:
|
private:
|
||||||
friend class ::nsAutoScrollTimer;
|
friend class ::nsAutoScrollTimer;
|
||||||
|
|
||||||
@ -229,6 +232,7 @@ public:
|
|||||||
AutoRestore<bool> mSavedValue;
|
AutoRestore<bool> mSavedValue;
|
||||||
MOZ_DECL_USE_GUARD_OBJECT_NOTIFIER
|
MOZ_DECL_USE_GUARD_OBJECT_NOTIFIER
|
||||||
};
|
};
|
||||||
|
|
||||||
private:
|
private:
|
||||||
friend struct mozilla::AutoPrepareFocusRange;
|
friend struct mozilla::AutoPrepareFocusRange;
|
||||||
class ScrollSelectionIntoViewEvent;
|
class ScrollSelectionIntoViewEvent;
|
||||||
@ -282,6 +286,8 @@ private:
|
|||||||
int32_t* aStartIndex, int32_t* aEndIndex);
|
int32_t* aStartIndex, int32_t* aEndIndex);
|
||||||
RangeData* FindRangeData(nsIDOMRange* aRange);
|
RangeData* FindRangeData(nsIDOMRange* aRange);
|
||||||
|
|
||||||
|
void UserSelectRangesToAdd(nsRange* aItem, nsTArray<nsRefPtr<nsRange> >& rangesToAdd);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Helper method for AddItem.
|
* Helper method for AddItem.
|
||||||
*/
|
*/
|
||||||
@ -312,9 +318,14 @@ private:
|
|||||||
SelectionType mType;
|
SelectionType mType;
|
||||||
/**
|
/**
|
||||||
* True if the current selection operation was initiated by user action.
|
* True if the current selection operation was initiated by user action.
|
||||||
* It determines whether we exclude -moz-user-select:none nodes or not.
|
* It determines whether we exclude -moz-user-select:none nodes or not,
|
||||||
|
* as well as whether selectstart events will be fired.
|
||||||
*/
|
*/
|
||||||
bool mApplyUserSelectStyle;
|
bool mApplyUserSelectStyle;
|
||||||
|
|
||||||
|
// Non-zero if we don't want any changes we make to the selection to be
|
||||||
|
// visible to content. If non-zero, content won't be notified about changes.
|
||||||
|
uint32_t mSelectionChangeBlockerCount;
|
||||||
};
|
};
|
||||||
|
|
||||||
// Stack-class to turn on/off selection batching.
|
// Stack-class to turn on/off selection batching.
|
||||||
@ -339,6 +350,33 @@ public:
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
class MOZ_STACK_CLASS AutoHideSelectionChanges final
|
||||||
|
{
|
||||||
|
private:
|
||||||
|
nsRefPtr<Selection> mSelection;
|
||||||
|
MOZ_DECL_USE_GUARD_OBJECT_NOTIFIER
|
||||||
|
public:
|
||||||
|
explicit AutoHideSelectionChanges(const nsFrameSelection* aFrame);
|
||||||
|
|
||||||
|
explicit AutoHideSelectionChanges(Selection* aSelection
|
||||||
|
MOZ_GUARD_OBJECT_NOTIFIER_PARAM)
|
||||||
|
: mSelection(aSelection)
|
||||||
|
{
|
||||||
|
MOZ_GUARD_OBJECT_NOTIFIER_INIT;
|
||||||
|
mSelection = aSelection;
|
||||||
|
if (mSelection) {
|
||||||
|
mSelection->AddSelectionChangeBlocker();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
~AutoHideSelectionChanges()
|
||||||
|
{
|
||||||
|
if (mSelection) {
|
||||||
|
mSelection->RemoveSelectionChangeBlocker();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
} // namespace dom
|
} // namespace dom
|
||||||
} // namespace mozilla
|
} // namespace mozilla
|
||||||
|
|
||||||
|
57
layout/generic/SelectionChangeListener.h
Normal file
57
layout/generic/SelectionChangeListener.h
Normal file
@ -0,0 +1,57 @@
|
|||||||
|
/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
|
||||||
|
/* 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_SelectionChangeListener_h_
|
||||||
|
#define mozilla_SelectionChangeListener_h_
|
||||||
|
|
||||||
|
#include "nsISelectionListener.h"
|
||||||
|
#include "nsISelectionPrivate.h"
|
||||||
|
#include "mozilla/Attributes.h"
|
||||||
|
|
||||||
|
namespace mozilla {
|
||||||
|
namespace dom {
|
||||||
|
|
||||||
|
class SelectionChangeListener final : public nsISelectionListener
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
// SelectionChangeListener has to participate in cycle collection because
|
||||||
|
// it holds strong references to nsINodes in its mOldRanges array.
|
||||||
|
NS_DECL_CYCLE_COLLECTING_ISUPPORTS
|
||||||
|
NS_DECL_CYCLE_COLLECTION_CLASS(SelectionChangeListener)
|
||||||
|
NS_DECL_NSISELECTIONLISTENER
|
||||||
|
|
||||||
|
// This field is used to keep track of the ranges which were present in the
|
||||||
|
// selection when the selectionchange event was previously fired. This allows
|
||||||
|
// for the selectionchange event to only be fired when a selection is actually
|
||||||
|
// changed.
|
||||||
|
struct RawRangeData
|
||||||
|
{
|
||||||
|
// These properties are not void*s to avoid the potential situation where the
|
||||||
|
// nsINode is freed, and a new nsINode is allocated with the same address, which
|
||||||
|
// could potentially break the comparison logic. In reality, this is extremely
|
||||||
|
// unlikely to occur (potentially impossible), but these nsCOMPtrs are safer.
|
||||||
|
// They are never dereferenced.
|
||||||
|
nsCOMPtr<nsINode> mStartParent;
|
||||||
|
nsCOMPtr<nsINode> mEndParent;
|
||||||
|
|
||||||
|
// XXX These are int32_ts on nsRange, but uint32_ts in the return value
|
||||||
|
// of GetStart_, so I use uint32_ts here. See bug 1194256.
|
||||||
|
uint32_t mStartOffset;
|
||||||
|
uint32_t mEndOffset;
|
||||||
|
|
||||||
|
explicit RawRangeData(const nsRange* aRange);
|
||||||
|
bool Equals(const nsRange* aRange);
|
||||||
|
};
|
||||||
|
|
||||||
|
private:
|
||||||
|
nsTArray<RawRangeData> mOldRanges;
|
||||||
|
|
||||||
|
~SelectionChangeListener() {}
|
||||||
|
};
|
||||||
|
|
||||||
|
} // namespace dom
|
||||||
|
} // namespace mozilla
|
||||||
|
|
||||||
|
#endif // mozilla_SelectionChangeListener_h_
|
@ -752,6 +752,8 @@ private:
|
|||||||
bool mDesiredPosSet;
|
bool mDesiredPosSet;
|
||||||
|
|
||||||
int8_t mCaretMovementStyle;
|
int8_t mCaretMovementStyle;
|
||||||
|
|
||||||
|
static bool sSelectionEventsEnabled;
|
||||||
};
|
};
|
||||||
|
|
||||||
#endif /* nsFrameSelection_h___ */
|
#endif /* nsFrameSelection_h___ */
|
||||||
|
@ -68,6 +68,7 @@ static NS_DEFINE_CID(kFrameTraversalCID, NS_FRAMETRAVERSAL_CID);
|
|||||||
|
|
||||||
#include "nsISelectionController.h"//for the enums
|
#include "nsISelectionController.h"//for the enums
|
||||||
#include "nsAutoCopyListener.h"
|
#include "nsAutoCopyListener.h"
|
||||||
|
#include "SelectionChangeListener.h"
|
||||||
#include "nsCopySupport.h"
|
#include "nsCopySupport.h"
|
||||||
#include "nsIClipboard.h"
|
#include "nsIClipboard.h"
|
||||||
#include "nsIFrameInlines.h"
|
#include "nsIFrameInlines.h"
|
||||||
@ -79,6 +80,7 @@ static NS_DEFINE_CID(kFrameTraversalCID, NS_FRAMETRAVERSAL_CID);
|
|||||||
#include "mozilla/dom/ShadowRoot.h"
|
#include "mozilla/dom/ShadowRoot.h"
|
||||||
#include "mozilla/ErrorResult.h"
|
#include "mozilla/ErrorResult.h"
|
||||||
#include "mozilla/dom/SelectionBinding.h"
|
#include "mozilla/dom/SelectionBinding.h"
|
||||||
|
#include "mozilla/AsyncEventDispatcher.h"
|
||||||
|
|
||||||
using namespace mozilla;
|
using namespace mozilla;
|
||||||
using namespace mozilla::dom;
|
using namespace mozilla::dom;
|
||||||
@ -412,7 +414,7 @@ struct MOZ_RAII AutoPrepareFocusRange
|
|||||||
while (i--) {
|
while (i--) {
|
||||||
range = aSelection->mRanges[i].mRange;
|
range = aSelection->mRanges[i].mRange;
|
||||||
if (range->IsGenerated()) {
|
if (range->IsGenerated()) {
|
||||||
range->SetInSelection(false);
|
range->SetSelection(nullptr);
|
||||||
aSelection->selectFrames(presContext, range, false);
|
aSelection->selectFrames(presContext, range, false);
|
||||||
aSelection->mRanges.RemoveElementAt(i);
|
aSelection->mRanges.RemoveElementAt(i);
|
||||||
}
|
}
|
||||||
@ -814,6 +816,16 @@ nsFrameSelection::Init(nsIPresShell *aShell, nsIContent *aLimiter)
|
|||||||
mLimiter = aLimiter;
|
mLimiter = aLimiter;
|
||||||
mCaretMovementStyle =
|
mCaretMovementStyle =
|
||||||
Preferences::GetInt("bidi.edit.caret_movement_style", 2);
|
Preferences::GetInt("bidi.edit.caret_movement_style", 2);
|
||||||
|
|
||||||
|
// This should only ever be initialized on the main thread, so we are OK here.
|
||||||
|
static bool prefCachesInitialized = false;
|
||||||
|
if (!prefCachesInitialized) {
|
||||||
|
prefCachesInitialized = true;
|
||||||
|
|
||||||
|
Preferences::AddBoolVarCache(&sSelectionEventsEnabled,
|
||||||
|
"dom.select_events.enabled", false);
|
||||||
|
}
|
||||||
|
|
||||||
// Set touch caret as selection listener
|
// Set touch caret as selection listener
|
||||||
nsRefPtr<TouchCaret> touchCaret = mShell->GetTouchCaret();
|
nsRefPtr<TouchCaret> touchCaret = mShell->GetTouchCaret();
|
||||||
if (touchCaret) {
|
if (touchCaret) {
|
||||||
@ -839,8 +851,21 @@ nsFrameSelection::Init(nsIPresShell *aShell, nsIContent *aLimiter)
|
|||||||
mDomSelections[index]->AddSelectionListener(eventHub);
|
mDomSelections[index]->AddSelectionListener(eventHub);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (sSelectionEventsEnabled) {
|
||||||
|
int8_t index =
|
||||||
|
GetIndexFromSelectionType(nsISelectionController::SELECTION_NORMAL);
|
||||||
|
if (mDomSelections[index]) {
|
||||||
|
// The Selection instance will hold a strong reference to its selectionchangelistener
|
||||||
|
// so we don't have to worry about that!
|
||||||
|
nsRefPtr<SelectionChangeListener> listener = new SelectionChangeListener;
|
||||||
|
mDomSelections[index]->AddSelectionListener(listener);
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool nsFrameSelection::sSelectionEventsEnabled = false;
|
||||||
|
|
||||||
nsresult
|
nsresult
|
||||||
nsFrameSelection::MoveCaret(nsDirection aDirection,
|
nsFrameSelection::MoveCaret(nsDirection aDirection,
|
||||||
bool aContinueSelection,
|
bool aContinueSelection,
|
||||||
@ -3307,6 +3332,7 @@ Selection::Selection()
|
|||||||
, mDirection(eDirNext)
|
, mDirection(eDirNext)
|
||||||
, mType(nsISelectionController::SELECTION_NORMAL)
|
, mType(nsISelectionController::SELECTION_NORMAL)
|
||||||
, mApplyUserSelectStyle(false)
|
, mApplyUserSelectStyle(false)
|
||||||
|
, mSelectionChangeBlockerCount(0)
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -3316,6 +3342,7 @@ Selection::Selection(nsFrameSelection* aList)
|
|||||||
, mDirection(eDirNext)
|
, mDirection(eDirNext)
|
||||||
, mType(nsISelectionController::SELECTION_NORMAL)
|
, mType(nsISelectionController::SELECTION_NORMAL)
|
||||||
, mApplyUserSelectStyle(false)
|
, mApplyUserSelectStyle(false)
|
||||||
|
, mSelectionChangeBlockerCount(0)
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -3325,7 +3352,7 @@ Selection::~Selection()
|
|||||||
|
|
||||||
uint32_t count = mRanges.Length();
|
uint32_t count = mRanges.Length();
|
||||||
for (uint32_t i = 0; i < count; ++i) {
|
for (uint32_t i = 0; i < count; ++i) {
|
||||||
mRanges[i].mRange->SetInSelection(false);
|
mRanges[i].mRange->SetSelection(nullptr);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (mAutoScrollTimer) {
|
if (mAutoScrollTimer) {
|
||||||
@ -3647,8 +3674,24 @@ Selection::SubtractRange(RangeData* aRange, nsRange* aSubtract,
|
|||||||
return NS_OK;
|
return NS_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
Selection::UserSelectRangesToAdd(nsRange* aItem, nsTArray<nsRefPtr<nsRange>>& aRangesToAdd)
|
||||||
|
{
|
||||||
|
aItem->ExcludeNonSelectableNodes(&aRangesToAdd);
|
||||||
|
if (aRangesToAdd.IsEmpty()) {
|
||||||
|
ErrorResult err;
|
||||||
|
nsINode* node = aItem->GetStartContainer(err);
|
||||||
|
if (node && node->IsContent() && node->AsContent()->GetEditingHost()) {
|
||||||
|
// A contenteditable node with user-select:none, for example.
|
||||||
|
// Allow it to have a collapsed selection (for the caret).
|
||||||
|
aItem->Collapse(GetDirection() == eDirPrevious);
|
||||||
|
aRangesToAdd.AppendElement(aItem);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
nsresult
|
nsresult
|
||||||
Selection::AddItem(nsRange* aItem, int32_t* aOutIndex)
|
Selection::AddItem(nsRange* aItem, int32_t* aOutIndex, bool aNoStartSelect)
|
||||||
{
|
{
|
||||||
if (!aItem)
|
if (!aItem)
|
||||||
return NS_ERROR_NULL_POINTER;
|
return NS_ERROR_NULL_POINTER;
|
||||||
@ -3657,20 +3700,54 @@ Selection::AddItem(nsRange* aItem, int32_t* aOutIndex)
|
|||||||
|
|
||||||
NS_ASSERTION(aOutIndex, "aOutIndex can't be null");
|
NS_ASSERTION(aOutIndex, "aOutIndex can't be null");
|
||||||
|
|
||||||
|
// XXX Rename mApplyUserSelectStyle? Not the best name (as it is also being
|
||||||
|
// used to detect here whether the event is user initiated for the purposes of
|
||||||
|
// dispatching the selectstart event).
|
||||||
if (mApplyUserSelectStyle) {
|
if (mApplyUserSelectStyle) {
|
||||||
nsAutoTArray<nsRefPtr<nsRange>, 4> rangesToAdd;
|
nsAutoTArray<nsRefPtr<nsRange>, 4> rangesToAdd;
|
||||||
aItem->ExcludeNonSelectableNodes(&rangesToAdd);
|
|
||||||
if (rangesToAdd.IsEmpty()) {
|
|
||||||
ErrorResult err;
|
|
||||||
nsINode* node = aItem->GetStartContainer(err);
|
|
||||||
if (node && node->IsContent() && node->AsContent()->GetEditingHost()) {
|
|
||||||
// A contenteditable node with user-select:none, for example.
|
|
||||||
// Allow it to have a collapsed selection (for the caret).
|
|
||||||
aItem->Collapse(GetDirection() == eDirPrevious);
|
|
||||||
rangesToAdd.AppendElement(aItem);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
*aOutIndex = -1;
|
*aOutIndex = -1;
|
||||||
|
|
||||||
|
if (!aNoStartSelect && mType == nsISelectionController::SELECTION_NORMAL &&
|
||||||
|
nsFrameSelection::sSelectionEventsEnabled && Collapsed() &&
|
||||||
|
!IsBlockingSelectionChangeEvents()) {
|
||||||
|
// First, we generate the ranges to add with a scratch range, which is a
|
||||||
|
// clone of the original range passed in. We do this seperately, because the
|
||||||
|
// selectstart event could have caused the world to change, and required
|
||||||
|
// ranges to be re-generated
|
||||||
|
nsRefPtr<nsRange> scratchRange = aItem->CloneRange();
|
||||||
|
UserSelectRangesToAdd(scratchRange, rangesToAdd);
|
||||||
|
bool newRangesNonEmpty = rangesToAdd.Length() > 1 ||
|
||||||
|
(rangesToAdd.Length() == 1 && !rangesToAdd[0]->Collapsed());
|
||||||
|
|
||||||
|
MOZ_ASSERT(!newRangesNonEmpty || nsContentUtils::IsSafeToRunScript());
|
||||||
|
if (newRangesNonEmpty && nsContentUtils::IsSafeToRunScript()) {
|
||||||
|
// We consider a selection to be starting if we are currently collapsed,
|
||||||
|
// and the selection is becoming uncollapsed, and this is caused by a user
|
||||||
|
// initiated event.
|
||||||
|
bool defaultAction = true;
|
||||||
|
nsContentUtils::DispatchTrustedEvent(GetParentObject(),
|
||||||
|
aItem->GetStartParent(),
|
||||||
|
NS_LITERAL_STRING("selectstart"),
|
||||||
|
true, true, &defaultAction);
|
||||||
|
|
||||||
|
if (!defaultAction) {
|
||||||
|
return NS_OK;
|
||||||
|
}
|
||||||
|
|
||||||
|
// As we just dispatched an event to the DOM, something could have
|
||||||
|
// changed under our feet. Re-generate the rangesToAdd array, and ensure
|
||||||
|
// that the range we are about to add is still valid.
|
||||||
|
if (!aItem->IsPositioned()) {
|
||||||
|
return NS_ERROR_UNEXPECTED;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// The scratch ranges we generated may be invalid now, throw them out
|
||||||
|
rangesToAdd.ClearAndRetainStorage();
|
||||||
|
}
|
||||||
|
|
||||||
|
// Generate the ranges to add
|
||||||
|
UserSelectRangesToAdd(aItem, rangesToAdd);
|
||||||
size_t newAnchorFocusIndex =
|
size_t newAnchorFocusIndex =
|
||||||
GetDirection() == eDirPrevious ? 0 : rangesToAdd.Length() - 1;
|
GetDirection() == eDirPrevious ? 0 : rangesToAdd.Length() - 1;
|
||||||
for (size_t i = 0; i < rangesToAdd.Length(); ++i) {
|
for (size_t i = 0; i < rangesToAdd.Length(); ++i) {
|
||||||
@ -3702,7 +3779,7 @@ Selection::AddItemInternal(nsRange* aItem, int32_t* aOutIndex)
|
|||||||
if (mRanges.Length() == 0) {
|
if (mRanges.Length() == 0) {
|
||||||
if (!mRanges.AppendElement(RangeData(aItem)))
|
if (!mRanges.AppendElement(RangeData(aItem)))
|
||||||
return NS_ERROR_OUT_OF_MEMORY;
|
return NS_ERROR_OUT_OF_MEMORY;
|
||||||
aItem->SetInSelection(true);
|
aItem->SetSelection(this);
|
||||||
|
|
||||||
*aOutIndex = 0;
|
*aOutIndex = 0;
|
||||||
return NS_OK;
|
return NS_OK;
|
||||||
@ -3742,7 +3819,7 @@ Selection::AddItemInternal(nsRange* aItem, int32_t* aOutIndex)
|
|||||||
// The new range doesn't overlap any existing ranges
|
// The new range doesn't overlap any existing ranges
|
||||||
if (!mRanges.InsertElementAt(startIndex, RangeData(aItem)))
|
if (!mRanges.InsertElementAt(startIndex, RangeData(aItem)))
|
||||||
return NS_ERROR_OUT_OF_MEMORY;
|
return NS_ERROR_OUT_OF_MEMORY;
|
||||||
aItem->SetInSelection(true);
|
aItem->SetSelection(this);
|
||||||
*aOutIndex = startIndex;
|
*aOutIndex = startIndex;
|
||||||
return NS_OK;
|
return NS_OK;
|
||||||
}
|
}
|
||||||
@ -3764,7 +3841,7 @@ Selection::AddItemInternal(nsRange* aItem, int32_t* aOutIndex)
|
|||||||
|
|
||||||
// Remove all the overlapping ranges
|
// Remove all the overlapping ranges
|
||||||
for (int32_t i = startIndex; i < endIndex; ++i) {
|
for (int32_t i = startIndex; i < endIndex; ++i) {
|
||||||
mRanges[i].mRange->SetInSelection(false);
|
mRanges[i].mRange->SetSelection(nullptr);
|
||||||
}
|
}
|
||||||
mRanges.RemoveElementsAt(startIndex, endIndex - startIndex);
|
mRanges.RemoveElementsAt(startIndex, endIndex - startIndex);
|
||||||
|
|
||||||
@ -3789,7 +3866,7 @@ Selection::AddItemInternal(nsRange* aItem, int32_t* aOutIndex)
|
|||||||
return NS_ERROR_OUT_OF_MEMORY;
|
return NS_ERROR_OUT_OF_MEMORY;
|
||||||
|
|
||||||
for (uint32_t i = 0; i < temp.Length(); ++i) {
|
for (uint32_t i = 0; i < temp.Length(); ++i) {
|
||||||
temp[i].mRange->SetInSelection(true);
|
temp[i].mRange->SetSelection(this);
|
||||||
}
|
}
|
||||||
|
|
||||||
*aOutIndex = startIndex + insertionPoint;
|
*aOutIndex = startIndex + insertionPoint;
|
||||||
@ -3818,7 +3895,7 @@ Selection::RemoveItem(nsRange* aItem)
|
|||||||
return NS_ERROR_INVALID_ARG;
|
return NS_ERROR_INVALID_ARG;
|
||||||
|
|
||||||
mRanges.RemoveElementAt(idx);
|
mRanges.RemoveElementAt(idx);
|
||||||
aItem->SetInSelection(false);
|
aItem->SetSelection(nullptr);
|
||||||
return NS_OK;
|
return NS_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -3843,7 +3920,7 @@ Selection::Clear(nsPresContext* aPresContext)
|
|||||||
setAnchorFocusRange(-1);
|
setAnchorFocusRange(-1);
|
||||||
|
|
||||||
for (uint32_t i = 0; i < mRanges.Length(); ++i) {
|
for (uint32_t i = 0; i < mRanges.Length(); ++i) {
|
||||||
mRanges[i].mRange->SetInSelection(false);
|
mRanges[i].mRange->SetSelection(nullptr);
|
||||||
selectFrames(aPresContext, mRanges[i].mRange, false);
|
selectFrames(aPresContext, mRanges[i].mRange, false);
|
||||||
}
|
}
|
||||||
mRanges.Clear();
|
mRanges.Clear();
|
||||||
@ -5064,12 +5141,14 @@ Selection::SetAnchorFocusToRange(nsRange* aRange)
|
|||||||
{
|
{
|
||||||
NS_ENSURE_STATE(mAnchorFocusRange);
|
NS_ENSURE_STATE(mAnchorFocusRange);
|
||||||
|
|
||||||
|
bool collapsed = Collapsed();
|
||||||
|
|
||||||
nsresult res = RemoveItem(mAnchorFocusRange);
|
nsresult res = RemoveItem(mAnchorFocusRange);
|
||||||
if (NS_FAILED(res))
|
if (NS_FAILED(res))
|
||||||
return res;
|
return res;
|
||||||
|
|
||||||
int32_t aOutIndex = -1;
|
int32_t aOutIndex = -1;
|
||||||
res = AddItem(aRange, &aOutIndex);
|
res = AddItem(aRange, &aOutIndex, !collapsed);
|
||||||
if (NS_FAILED(res))
|
if (NS_FAILED(res))
|
||||||
return res;
|
return res;
|
||||||
setAnchorFocusRange(aOutIndex);
|
setAnchorFocusRange(aOutIndex);
|
||||||
@ -5471,19 +5550,16 @@ Selection::SelectAllChildren(nsIDOMNode* aParentNode)
|
|||||||
void
|
void
|
||||||
Selection::SelectAllChildren(nsINode& aNode, ErrorResult& aRv)
|
Selection::SelectAllChildren(nsINode& aNode, ErrorResult& aRv)
|
||||||
{
|
{
|
||||||
if (mFrameSelection)
|
if (mFrameSelection) {
|
||||||
{
|
|
||||||
mFrameSelection->PostReason(nsISelectionListener::SELECTALL_REASON);
|
mFrameSelection->PostReason(nsISelectionListener::SELECTALL_REASON);
|
||||||
}
|
}
|
||||||
|
SelectionBatcher batch(this);
|
||||||
|
|
||||||
Collapse(aNode, 0, aRv);
|
Collapse(aNode, 0, aRv);
|
||||||
if (aRv.Failed()) {
|
if (aRv.Failed()) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (mFrameSelection)
|
|
||||||
{
|
|
||||||
mFrameSelection->PostReason(nsISelectionListener::SELECTALL_REASON);
|
|
||||||
}
|
|
||||||
Extend(aNode, aNode.GetChildCount(), aRv);
|
Extend(aNode, aNode.GetChildCount(), aRv);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -5926,7 +6002,26 @@ Selection::EndBatchChanges()
|
|||||||
return NS_OK;
|
return NS_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
Selection::AddSelectionChangeBlocker()
|
||||||
|
{
|
||||||
|
mSelectionChangeBlockerCount++;
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
Selection::RemoveSelectionChangeBlocker()
|
||||||
|
{
|
||||||
|
MOZ_ASSERT(mSelectionChangeBlockerCount > 0,
|
||||||
|
"mSelectionChangeBlockerCount has an invalid value - "
|
||||||
|
"maybe you have a mismatched RemoveSelectionChangeBlocker?");
|
||||||
|
mSelectionChangeBlockerCount--;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool
|
||||||
|
Selection::IsBlockingSelectionChangeEvents() const
|
||||||
|
{
|
||||||
|
return mSelectionChangeBlockerCount > 0;
|
||||||
|
}
|
||||||
|
|
||||||
NS_IMETHODIMP
|
NS_IMETHODIMP
|
||||||
Selection::DeleteFromDocument()
|
Selection::DeleteFromDocument()
|
||||||
@ -6159,6 +6254,11 @@ Selection::WrapObject(JSContext* aCx, JS::Handle<JSObject*> aGivenProto)
|
|||||||
return mozilla::dom::SelectionBinding::Wrap(aCx, this, aGivenProto);
|
return mozilla::dom::SelectionBinding::Wrap(aCx, this, aGivenProto);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// AutoHideSelectionChanges
|
||||||
|
AutoHideSelectionChanges::AutoHideSelectionChanges(const nsFrameSelection* aFrame)
|
||||||
|
: AutoHideSelectionChanges(aFrame->GetSelection(nsISelectionController::SELECTION_NORMAL))
|
||||||
|
{}
|
||||||
|
|
||||||
// nsAutoCopyListener
|
// nsAutoCopyListener
|
||||||
|
|
||||||
nsAutoCopyListener* nsAutoCopyListener::sInstance = nullptr;
|
nsAutoCopyListener* nsAutoCopyListener::sInstance = nullptr;
|
||||||
@ -6218,3 +6318,103 @@ nsAutoCopyListener::NotifySelectionChanged(nsIDOMDocument *aDoc,
|
|||||||
return nsCopySupport::HTMLCopy(aSel, doc,
|
return nsCopySupport::HTMLCopy(aSel, doc,
|
||||||
nsIClipboard::kSelectionClipboard, false);
|
nsIClipboard::kSelectionClipboard, false);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// SelectionChangeListener
|
||||||
|
|
||||||
|
SelectionChangeListener::RawRangeData::RawRangeData(const nsRange* aRange)
|
||||||
|
{
|
||||||
|
mozilla::ErrorResult rv;
|
||||||
|
mStartParent = aRange->GetStartContainer(rv);
|
||||||
|
rv.SuppressException();
|
||||||
|
mEndParent = aRange->GetEndContainer(rv);
|
||||||
|
rv.SuppressException();
|
||||||
|
mStartOffset = aRange->GetStartOffset(rv);
|
||||||
|
rv.SuppressException();
|
||||||
|
mEndOffset = aRange->GetEndOffset(rv);
|
||||||
|
rv.SuppressException();
|
||||||
|
}
|
||||||
|
|
||||||
|
bool
|
||||||
|
SelectionChangeListener::RawRangeData::Equals(const nsRange* aRange)
|
||||||
|
{
|
||||||
|
mozilla::ErrorResult rv;
|
||||||
|
bool eq = mStartParent == aRange->GetStartContainer(rv);
|
||||||
|
rv.SuppressException();
|
||||||
|
eq = eq && mEndParent == aRange->GetEndContainer(rv);
|
||||||
|
rv.SuppressException();
|
||||||
|
eq = eq && mStartOffset == aRange->GetStartOffset(rv);
|
||||||
|
rv.SuppressException();
|
||||||
|
eq = eq && mEndOffset == aRange->GetEndOffset(rv);
|
||||||
|
rv.SuppressException();
|
||||||
|
return eq;
|
||||||
|
}
|
||||||
|
|
||||||
|
inline void
|
||||||
|
ImplCycleCollectionTraverse(nsCycleCollectionTraversalCallback& aCallback,
|
||||||
|
SelectionChangeListener::RawRangeData& aField,
|
||||||
|
const char* aName,
|
||||||
|
uint32_t aFlags = 0)
|
||||||
|
{
|
||||||
|
ImplCycleCollectionTraverse(aCallback, aField.mStartParent, "mStartParent", aFlags);
|
||||||
|
ImplCycleCollectionTraverse(aCallback, aField.mEndParent, "mEndParent", aFlags);
|
||||||
|
}
|
||||||
|
|
||||||
|
NS_IMPL_CYCLE_COLLECTION_CLASS(SelectionChangeListener)
|
||||||
|
|
||||||
|
NS_IMPL_CYCLE_COLLECTION_UNLINK_BEGIN(SelectionChangeListener)
|
||||||
|
tmp->mOldRanges.Clear();
|
||||||
|
NS_IMPL_CYCLE_COLLECTION_UNLINK_END
|
||||||
|
|
||||||
|
NS_IMPL_CYCLE_COLLECTION_TRAVERSE_BEGIN(SelectionChangeListener)
|
||||||
|
NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mOldRanges);
|
||||||
|
NS_IMPL_CYCLE_COLLECTION_TRAVERSE_END
|
||||||
|
|
||||||
|
NS_INTERFACE_MAP_BEGIN_CYCLE_COLLECTION(SelectionChangeListener)
|
||||||
|
NS_INTERFACE_MAP_ENTRY(nsISupports)
|
||||||
|
NS_INTERFACE_MAP_ENTRY(nsISelectionListener)
|
||||||
|
NS_INTERFACE_MAP_END
|
||||||
|
|
||||||
|
NS_IMPL_CYCLE_COLLECTING_ADDREF(SelectionChangeListener)
|
||||||
|
NS_IMPL_CYCLE_COLLECTING_RELEASE(SelectionChangeListener)
|
||||||
|
|
||||||
|
|
||||||
|
NS_IMETHODIMP
|
||||||
|
SelectionChangeListener::NotifySelectionChanged(nsIDOMDocument* aDoc,
|
||||||
|
nsISelection* aSel, int16_t aReason)
|
||||||
|
{
|
||||||
|
// This cast is valid as nsISelection is a builtinclass which is only
|
||||||
|
// implemented by Selection.
|
||||||
|
nsRefPtr<Selection> sel = static_cast<Selection*>(aSel);
|
||||||
|
|
||||||
|
// Check if the ranges have actually changed
|
||||||
|
if (mOldRanges.Length() == sel->RangeCount()) {
|
||||||
|
bool changed = false;
|
||||||
|
|
||||||
|
for (size_t i = 0; i < mOldRanges.Length(); i++) {
|
||||||
|
if (!mOldRanges[i].Equals(sel->GetRangeAt(i))) {
|
||||||
|
changed = true;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!changed) {
|
||||||
|
return NS_OK;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// The ranges have actually changed, update the mOldRanges array
|
||||||
|
mOldRanges.ClearAndRetainStorage();
|
||||||
|
for (size_t i = 0; i < sel->RangeCount(); i++) {
|
||||||
|
mOldRanges.AppendElement(RawRangeData(sel->GetRangeAt(i)));
|
||||||
|
}
|
||||||
|
|
||||||
|
// Actually fire off the event
|
||||||
|
nsCOMPtr<nsIDocument> doc = do_QueryInterface(aDoc);
|
||||||
|
if (doc) {
|
||||||
|
nsRefPtr<AsyncEventDispatcher> asyncDispatcher =
|
||||||
|
new AsyncEventDispatcher(doc, NS_LITERAL_STRING("selectionchange"), false);
|
||||||
|
asyncDispatcher->PostDOMEvent();
|
||||||
|
}
|
||||||
|
|
||||||
|
return NS_OK;
|
||||||
|
}
|
||||||
|
@ -134,6 +134,13 @@ pref("dom.permissions.enabled", true);
|
|||||||
pref("dom.permissions.enabled", false);
|
pref("dom.permissions.enabled", false);
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
// Whether or not selection events are enabled
|
||||||
|
#ifdef NIGHTLY_BUILD
|
||||||
|
pref("dom.select_events.enabled", true);
|
||||||
|
#else
|
||||||
|
pref("dom.select_events.enabled", false);
|
||||||
|
#endif
|
||||||
|
|
||||||
// Whether or not Web Workers are enabled.
|
// Whether or not Web Workers are enabled.
|
||||||
pref("dom.workers.enabled", true);
|
pref("dom.workers.enabled", true);
|
||||||
// The number of workers per domain allowed to run concurrently.
|
// The number of workers per domain allowed to run concurrently.
|
||||||
|
@ -424,3 +424,8 @@ NS_EVENT_MESSAGE(eGamepadEventLast, eGamepadDisconnected)
|
|||||||
// input and beforeinput events.
|
// input and beforeinput events.
|
||||||
NS_EVENT_MESSAGE(NS_EDITOR_EVENT_START, 6100)
|
NS_EVENT_MESSAGE(NS_EDITOR_EVENT_START, 6100)
|
||||||
NS_EVENT_MESSAGE(NS_EDITOR_INPUT, NS_EDITOR_EVENT_START)
|
NS_EVENT_MESSAGE(NS_EDITOR_INPUT, NS_EDITOR_EVENT_START)
|
||||||
|
|
||||||
|
// selection events
|
||||||
|
NS_EVENT_MESSAGE(NS_SELECT_EVENT_START, 6200)
|
||||||
|
NS_EVENT_MESSAGE(NS_SELECT_START, NS_SELECT_EVENT_START)
|
||||||
|
NS_EVENT_MESSAGE(NS_SELECTION_CHANGE, NS_SELECT_EVENT_START + 1)
|
||||||
|
Loading…
x
Reference in New Issue
Block a user