1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263
|
/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
/* vim: set ts=8 sts=2 et sw=2 tw=80: */
/* This Source Code Form is subject to the terms of the Mozilla Public
* License, v. 2.0. If a copy of the MPL was not distributed with this
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
#include "mozilla/dom/CrossShadowBoundaryRange.h"
#include "nsContentUtils.h"
#include "nsIContentInlines.h"
#include "nsINode.h"
#include "nsRange.h"
namespace mozilla::dom {
template already_AddRefed<CrossShadowBoundaryRange>
CrossShadowBoundaryRange::Create(const RangeBoundary& aStartBoundary,
const RangeBoundary& aEndBoundary,
nsRange* aOwner);
template already_AddRefed<CrossShadowBoundaryRange>
CrossShadowBoundaryRange::Create(const RangeBoundary& aStartBoundary,
const RawRangeBoundary& aEndBoundary,
nsRange* aOwner);
template already_AddRefed<CrossShadowBoundaryRange>
CrossShadowBoundaryRange::Create(const RawRangeBoundary& aStartBoundary,
const RangeBoundary& aEndBoundary,
nsRange* aOwner);
template already_AddRefed<CrossShadowBoundaryRange>
CrossShadowBoundaryRange::Create(const RawRangeBoundary& aStartBoundary,
const RawRangeBoundary& aEndBoundary,
nsRange* aOwner);
template void CrossShadowBoundaryRange::DoSetRange(
const RangeBoundary& aStartBoundary, const RangeBoundary& aEndBoundary,
nsINode* aRootNode, nsRange* aOwner);
template void CrossShadowBoundaryRange::DoSetRange(
const RangeBoundary& aStartBoundary, const RawRangeBoundary& aEndBoundary,
nsINode* aRootNode, nsRange* aOwner);
template void CrossShadowBoundaryRange::DoSetRange(
const RawRangeBoundary& aStartBoundary, const RangeBoundary& aEndBoundary,
nsINode* aRootNode, nsRange* aOwner);
template void CrossShadowBoundaryRange::DoSetRange(
const RawRangeBoundary& aStartBoundary,
const RawRangeBoundary& aEndBoundary, nsINode* aRootNode, nsRange* aOwner);
template nsresult CrossShadowBoundaryRange::SetStartAndEnd(
const RangeBoundary& aStartBoundary, const RangeBoundary& aEndBoundary);
template nsresult CrossShadowBoundaryRange::SetStartAndEnd(
const RangeBoundary& aStartBoundary, const RawRangeBoundary& aEndBoundary);
template nsresult CrossShadowBoundaryRange::SetStartAndEnd(
const RawRangeBoundary& aStartBoundary, const RangeBoundary& aEndBoundary);
template nsresult CrossShadowBoundaryRange::SetStartAndEnd(
const RawRangeBoundary& aStartBoundary,
const RawRangeBoundary& aEndBoundary);
nsTArray<RefPtr<CrossShadowBoundaryRange>>*
CrossShadowBoundaryRange::sCachedRanges = nullptr;
NS_IMPL_CYCLE_COLLECTING_ADDREF(CrossShadowBoundaryRange)
NS_IMPL_CYCLE_COLLECTING_RELEASE_WITH_INTERRUPTABLE_LAST_RELEASE(
CrossShadowBoundaryRange,
DoSetRange(RawRangeBoundary(TreeKind::Flat),
RawRangeBoundary(TreeKind::Flat), nullptr, nullptr),
AbstractRange::MaybeCacheToReuse(*this))
NS_INTERFACE_MAP_BEGIN_CYCLE_COLLECTION(CrossShadowBoundaryRange)
NS_INTERFACE_MAP_END_INHERITING(CrossShadowBoundaryRange)
NS_IMPL_CYCLE_COLLECTION_CLASS(CrossShadowBoundaryRange)
NS_IMPL_CYCLE_COLLECTION_UNLINK_BEGIN_INHERITED(CrossShadowBoundaryRange,
StaticRange)
if (tmp->mCommonAncestor) {
tmp->mCommonAncestor->RemoveMutationObserver(tmp);
}
NS_IMPL_CYCLE_COLLECTION_UNLINK(mCommonAncestor)
NS_IMPL_CYCLE_COLLECTION_UNLINK_END
NS_IMPL_CYCLE_COLLECTION_TRAVERSE_BEGIN_INHERITED(CrossShadowBoundaryRange,
StaticRange)
NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mCommonAncestor)
NS_IMPL_CYCLE_COLLECTION_TRAVERSE_END
NS_IMPL_CYCLE_COLLECTION_TRACE_BEGIN_INHERITED(CrossShadowBoundaryRange,
StaticRange)
NS_IMPL_CYCLE_COLLECTION_TRACE_END
/* static */
template <typename SPT, typename SRT, typename EPT, typename ERT>
already_AddRefed<CrossShadowBoundaryRange> CrossShadowBoundaryRange::Create(
const RangeBoundaryBase<SPT, SRT>& aStartBoundary,
const RangeBoundaryBase<EPT, ERT>& aEndBoundary, nsRange* aOwner) {
RefPtr<CrossShadowBoundaryRange> range;
if (!sCachedRanges || sCachedRanges->IsEmpty()) {
range = new CrossShadowBoundaryRange(aStartBoundary.GetContainer(), aOwner);
} else {
range = sCachedRanges->PopLastElement().forget();
}
range->Init(aStartBoundary.GetContainer());
range->DoSetRange(aStartBoundary, aEndBoundary, nullptr, aOwner);
return range.forget();
}
template <typename SPT, typename SRT, typename EPT, typename ERT>
void CrossShadowBoundaryRange::DoSetRange(
const RangeBoundaryBase<SPT, SRT>& aStartBoundary,
const RangeBoundaryBase<EPT, ERT>& aEndBoundary, nsINode* aRootNode,
nsRange* aOwner) {
// aRootNode is useless to CrossShadowBoundaryRange because aStartBoundary
// and aEndBoundary could have different roots.
StaticRange::DoSetRange(aStartBoundary, aEndBoundary, nullptr);
nsINode* startRoot = RangeUtils::ComputeRootNode(mStart.GetContainer());
nsINode* endRoot = RangeUtils::ComputeRootNode(mEnd.GetContainer());
nsINode* previousCommonAncestor = mCommonAncestor;
mCommonAncestor =
startRoot == endRoot
? startRoot
: nsContentUtils::GetClosestCommonShadowIncludingInclusiveAncestor(
mStart.GetContainer(), mEnd.GetContainer());
MOZ_ASSERT_IF(mOwner, mOwner == aOwner || !aOwner);
mOwner = aOwner;
if (previousCommonAncestor != mCommonAncestor) {
if (previousCommonAncestor) {
previousCommonAncestor->RemoveMutationObserver(this);
}
if (mCommonAncestor) {
mCommonAncestor->AddMutationObserver(this);
}
}
}
void CrossShadowBoundaryRange::ContentWillBeRemoved(nsIContent* aChild,
const ContentRemoveInfo&) {
// It's unclear from the spec about what should the selection be after
// DOM mutation. See https://github.com/w3c/selection-api/issues/168
//
// For now, we just clear the selection if the removed node is related
// to mStart or mEnd.
MOZ_DIAGNOSTIC_ASSERT(mOwner);
MOZ_DIAGNOSTIC_ASSERT(mOwner->GetCrossShadowBoundaryRange() == this);
RefPtr<CrossShadowBoundaryRange> kungFuDeathGrip(this);
const nsINode* startContainer = mStart.GetContainer();
const nsINode* endContainer = mEnd.GetContainer();
MOZ_ASSERT(startContainer && endContainer);
if (startContainer == aChild || endContainer == aChild) {
mOwner->ResetCrossShadowBoundaryRange();
return;
}
// This is a special case that the startContainer and endContainer could
// anonymous contents created by the frame of aChild, and they are
// unbounded from the document now.
if (!startContainer->IsInComposedDoc() || !endContainer->IsInComposedDoc()) {
mOwner->ResetCrossShadowBoundaryRange();
return;
}
if (const auto* shadowRoot = aChild->GetShadowRoot()) {
if (startContainer == shadowRoot || endContainer == shadowRoot) {
mOwner->ResetCrossShadowBoundaryRange();
return;
}
}
if (startContainer->IsShadowIncludingInclusiveDescendantOf(aChild) ||
endContainer->IsShadowIncludingInclusiveDescendantOf(aChild)) {
mOwner->ResetCrossShadowBoundaryRange();
return;
}
nsINode* container = aChild->GetParentNode();
auto MaybeCreateNewBoundary =
[container, aChild](
const nsINode* aContainer,
const RangeBoundary& aBoundary) -> Maybe<RawRangeBoundary> {
if (container == aContainer) {
// We're only interested if our boundary reference was removed, otherwise
// we can just invalidate the offset.
if (aChild == aBoundary.Ref()) {
return Some<RawRangeBoundary>(
{container, aChild->GetPreviousSibling(), TreeKind::Flat});
}
RawRangeBoundary newBoundary(TreeKind::Flat);
newBoundary.CopyFrom(aBoundary, RangeBoundaryIsMutationObserved::Yes);
newBoundary.InvalidateOffset();
return Some(newBoundary);
}
return Nothing();
};
const Maybe<RawRangeBoundary> newStartBoundary =
MaybeCreateNewBoundary(startContainer, mStart);
const Maybe<RawRangeBoundary> newEndBoundary =
MaybeCreateNewBoundary(endContainer, mEnd);
if (newStartBoundary || newEndBoundary) {
DoSetRange(newStartBoundary ? newStartBoundary.ref() : mStart.AsRaw(),
newEndBoundary ? newEndBoundary.ref() : mEnd.AsRaw(), nullptr,
mOwner);
}
}
// For now CrossShadowBoundaryRange::CharacterDataChanged is only meant
// to handle the character removal initiated by nsRange::CutContents.
void CrossShadowBoundaryRange::CharacterDataChanged(
nsIContent* aContent, const CharacterDataChangeInfo& aInfo) {
// When aInfo.mDetails is present, it means the character data was
// changed due to splitText() or normalize(), which shouldn't be the
// case for nsRange::CutContents, so we return early.
if (aInfo.mDetails) {
return;
}
MOZ_ASSERT(aContent);
MOZ_ASSERT(mIsPositioned);
auto MaybeCreateNewBoundary =
[aContent,
&aInfo](const RangeBoundary& aBoundary) -> Maybe<RawRangeBoundary> {
// If the changed node contains our start boundary and the change starts
// before the boundary we'll need to adjust the offset.
if (aContent == aBoundary.GetContainer() &&
// aInfo.mChangeStart is the offset where the change starts, if it's
// smaller than the offset of aBoundary, it means the characters
// before the selected content is changed (i.e, removed), so the
// offset of aBoundary needs to be adjusted.
aInfo.mChangeStart <
*aBoundary.Offset(
RangeBoundary::OffsetFilter::kValidOrInvalidOffsets)) {
RawRangeBoundary newStart =
nsRange::ComputeNewBoundaryWhenBoundaryInsideChangedText(
aInfo, aBoundary.AsRaw());
return Some(newStart.AsRangeBoundaryInFlatTree());
}
return Nothing();
};
const Maybe<RawRangeBoundary> newStartBoundary =
MaybeCreateNewBoundary(mStart);
const Maybe<RawRangeBoundary> newEndBoundary = MaybeCreateNewBoundary(mEnd);
if (newStartBoundary || newEndBoundary) {
DoSetRange(newStartBoundary ? newStartBoundary.ref() : mStart.AsRaw(),
newEndBoundary ? newEndBoundary.ref() : mEnd.AsRaw(), nullptr,
mOwner);
}
}
// DOM mutation for shadow-crossing selection is not specified.
// Spec issue: https://github.com/w3c/selection-api/issues/168
void CrossShadowBoundaryRange::ParentChainChanged(nsIContent* aContent) {
MOZ_DIAGNOSTIC_ASSERT(mCommonAncestor == aContent,
"Wrong ParentChainChanged notification");
MOZ_DIAGNOSTIC_ASSERT(mOwner);
mOwner->ResetCrossShadowBoundaryRange();
}
} // namespace mozilla::dom
|