gecko-dev/layout/painting/DisplayListClipState.cpp
Markus Stange 6efd7c73b1 Bug 1298218 - Use DisplayItemClipChain for tracking clips on display items. r=mattwoodrow,tnikkel
This is the bulk of the changes.
 - DisplayItemScrollClip is removed. Instead, we will have 1) ActiveScrolledRoot
   and 2) DisplayItemClipChain.
 - ActiveScrolledRoot points to a scroll frame and allows traversing up the
   scroll frame chain.
 - DisplayItemClipChain is a linked list of clips, each clip being associated
   with the ActiveScrolledRoot that moves this clip.
 - Each display item has an ActiveScrolledRoot and a clip chain.
 - nsDisplayItem::GetClip returns the item of the clip chain that scrolls with
   the item's ASR. The separation between "regular clip" and "scroll clips"
   mostly goes away.
 - Tracking clips in the display list builder's clip state happens very
   similarly to how regular clips used to be tracked - there's a clip chain for
   content descendants and a clip chain for containing block descendants. These
   clip chains are intersected to create the combined clip chain.
 - There are strict rules for the ASR of a container item: A container item's
   ASR should be the innermost ASR which the item has finite clipped bounds with
   respect to.
 - At some point in the future, ASRs and AGRs should be reunified, but I haven't
   done that yet, because I needed to limit the scope of the change.

MozReview-Commit-ID: KYEpWY7qgf2

--HG--
extra : rebase_source : c727f6300a35463750639e165bfa37374c06b851
2017-01-31 17:07:35 -05:00

172 lines
6.6 KiB
C++

/* -*- Mode: C++; tab-width: 20; 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/. */
#include "DisplayListClipState.h"
#include "nsDisplayList.h"
namespace mozilla {
void
DisplayListClipState::ClearUpToASR(const ActiveScrolledRoot* aASR)
{
while (mClipChainContentDescendants &&
ActiveScrolledRoot::IsAncestor(aASR, mClipChainContentDescendants->mASR)) {
mClipChainContentDescendants = mClipChainContentDescendants->mParent;
}
while (mClipChainContainingBlockDescendants &&
ActiveScrolledRoot::IsAncestor(aASR, mClipChainContainingBlockDescendants->mASR)) {
mClipChainContainingBlockDescendants = mClipChainContainingBlockDescendants->mParent;
}
InvalidateCurrentCombinedClipChain(aASR);
}
const DisplayItemClipChain*
DisplayListClipState::GetCurrentCombinedClipChain(nsDisplayListBuilder* aBuilder)
{
if (mCurrentCombinedClipChainIsValid) {
return mCurrentCombinedClipChain;
}
if (!mClipChainContentDescendants && !mClipChainContainingBlockDescendants) {
mCurrentCombinedClipChain = nullptr;
mCurrentCombinedClipChainIsValid = true;
return nullptr;
}
mCurrentCombinedClipChain =
aBuilder->CreateClipChainIntersection(mCurrentCombinedClipChain,
mClipChainContentDescendants,
mClipChainContainingBlockDescendants);
mCurrentCombinedClipChainIsValid = true;
return mCurrentCombinedClipChain;
}
static void
ApplyClip(nsDisplayListBuilder* aBuilder,
const DisplayItemClipChain*& aClipToModify,
const ActiveScrolledRoot* aASR,
DisplayItemClipChain& aClipChainOnStack)
{
aClipChainOnStack.mASR = aASR;
if (aClipToModify && aClipToModify->mASR == aASR) {
// Intersect with aClipToModify and replace the clip chain item.
aClipChainOnStack.mClip.IntersectWith(aClipToModify->mClip);
aClipChainOnStack.mParent = aClipToModify->mParent;
aClipToModify = &aClipChainOnStack;
} else if (!aClipToModify ||
ActiveScrolledRoot::IsAncestor(aClipToModify->mASR, aASR)) {
// Add a new clip chain item at the bottom.
aClipChainOnStack.mParent = aClipToModify;
aClipToModify = &aClipChainOnStack;
} else {
// We need to insert / intersect a DisplayItemClipChain in the middle of the
// aClipToModify chain. This is a very rare case.
// Find the common ancestor and have the builder create the DisplayItemClipChain
// intersection. This will create new DisplayItemClipChain objects for all
// descendants of ancestorSC and we will not hold on to a pointer to
// aClipChainOnStack.
const DisplayItemClipChain* ancestorSC = aClipToModify;
while (ancestorSC && ActiveScrolledRoot::IsAncestor(aASR, ancestorSC->mASR)) {
ancestorSC = ancestorSC->mParent;
}
aClipChainOnStack.mParent = nullptr;
aClipToModify =
aBuilder->CreateClipChainIntersection(ancestorSC, aClipToModify, &aClipChainOnStack);
}
}
void
DisplayListClipState::ClipContainingBlockDescendants(nsDisplayListBuilder* aBuilder,
const nsRect& aRect,
const nscoord* aRadii,
DisplayItemClipChain& aClipChainOnStack)
{
if (aRadii) {
aClipChainOnStack.mClip.SetTo(aRect, aRadii);
} else {
aClipChainOnStack.mClip.SetTo(aRect);
}
const ActiveScrolledRoot* asr = aBuilder->CurrentActiveScrolledRoot();
ApplyClip(aBuilder, mClipChainContainingBlockDescendants, asr, aClipChainOnStack);
InvalidateCurrentCombinedClipChain(asr);
}
void
DisplayListClipState::ClipContentDescendants(nsDisplayListBuilder* aBuilder,
const nsRect& aRect,
const nscoord* aRadii,
DisplayItemClipChain& aClipChainOnStack)
{
if (aRadii) {
aClipChainOnStack.mClip.SetTo(aRect, aRadii);
} else {
aClipChainOnStack.mClip.SetTo(aRect);
}
const ActiveScrolledRoot* asr = aBuilder->CurrentActiveScrolledRoot();
ApplyClip(aBuilder, mClipChainContentDescendants, asr, aClipChainOnStack);
InvalidateCurrentCombinedClipChain(asr);
}
void
DisplayListClipState::ClipContentDescendants(nsDisplayListBuilder* aBuilder,
const nsRect& aRect,
const nsRect& aRoundedRect,
const nscoord* aRadii,
DisplayItemClipChain& aClipChainOnStack)
{
if (aRadii) {
aClipChainOnStack.mClip.SetTo(aRect, aRoundedRect, aRadii);
} else {
nsRect intersect = aRect.Intersect(aRoundedRect);
aClipChainOnStack.mClip.SetTo(intersect);
}
const ActiveScrolledRoot* asr = aBuilder->CurrentActiveScrolledRoot();
ApplyClip(aBuilder, mClipChainContentDescendants, asr, aClipChainOnStack);
InvalidateCurrentCombinedClipChain(asr);
}
void
DisplayListClipState::InvalidateCurrentCombinedClipChain(const ActiveScrolledRoot* aInvalidateUpTo)
{
mCurrentCombinedClipChainIsValid = false;
while (mCurrentCombinedClipChain &&
ActiveScrolledRoot::IsAncestor(aInvalidateUpTo, mCurrentCombinedClipChain->mASR)) {
mCurrentCombinedClipChain = mCurrentCombinedClipChain->mParent;
}
}
void
DisplayListClipState::ClipContainingBlockDescendantsToContentBox(nsDisplayListBuilder* aBuilder,
nsIFrame* aFrame,
DisplayItemClipChain& aClipChainOnStack,
uint32_t aFlags)
{
nscoord radii[8];
bool hasBorderRadius = aFrame->GetContentBoxBorderRadii(radii);
if (!hasBorderRadius && (aFlags & ASSUME_DRAWING_RESTRICTED_TO_CONTENT_RECT)) {
return;
}
nsRect clipRect = aFrame->GetContentRectRelativeToSelf() +
aBuilder->ToReferenceFrame(aFrame);
// If we have a border-radius, we have to clip our content to that
// radius.
ClipContainingBlockDescendants(aBuilder, clipRect, hasBorderRadius ? radii : nullptr,
aClipChainOnStack);
}
DisplayListClipState::AutoSaveRestore::AutoSaveRestore(nsDisplayListBuilder* aBuilder)
: mBuilder(aBuilder)
, mState(aBuilder->ClipState())
, mSavedState(aBuilder->ClipState())
#ifdef DEBUG
, mClipUsed(false)
, mRestored(false)
#endif
{}
} // namespace mozilla