This class is a RAII class that can be used to push a stacking context with properties from a WebRenderLayer. It can also then be used to convert rects in the layer coordinate system to be relative to the stacking context, which is what we want for passing to WR. MozReview-Commit-ID: 1WVrfRYqLqc
152 lines
4.5 KiB
C++
152 lines
4.5 KiB
C++
/* -*- 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/. */
|
|
|
|
#include "WebRenderPaintedLayer.h"
|
|
|
|
#include "LayersLogging.h"
|
|
#include "mozilla/ArrayUtils.h"
|
|
#include "mozilla/layers/StackingContextHelper.h"
|
|
#include "mozilla/layers/WebRenderBridgeChild.h"
|
|
#include "mozilla/layers/UpdateImageHelper.h"
|
|
#include "mozilla/webrender/WebRenderTypes.h"
|
|
#include "gfxPrefs.h"
|
|
#include "gfxUtils.h"
|
|
|
|
namespace mozilla {
|
|
namespace layers {
|
|
|
|
using namespace mozilla::gfx;
|
|
|
|
bool
|
|
WebRenderPaintedLayer::SetupExternalImages()
|
|
{
|
|
// XXX We won't keep using ContentClient for WebRenderPaintedLayer in the future and
|
|
// there is a crash problem for ContentClient on MacOS. So replace ContentClient with
|
|
// ImageClient. See bug 1341001.
|
|
|
|
if (!mImageContainer) {
|
|
mImageContainer = LayerManager::CreateImageContainer();
|
|
}
|
|
|
|
if (!mImageClient) {
|
|
mImageClient = ImageClient::CreateImageClient(CompositableType::IMAGE,
|
|
WrBridge(),
|
|
TextureFlags::DEFAULT);
|
|
if (!mImageClient) {
|
|
return false;
|
|
}
|
|
mImageClient->Connect();
|
|
}
|
|
|
|
if (mExternalImageId.isNothing()) {
|
|
mExternalImageId = Some(WrBridge()->AllocExternalImageIdForCompositable(mImageClient));
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
bool
|
|
WebRenderPaintedLayer::UpdateImageClient()
|
|
{
|
|
MOZ_ASSERT(Manager()->GetPaintedLayerCallback());
|
|
LayerIntRegion visibleRegion = GetVisibleRegion();
|
|
LayerIntRect bounds = visibleRegion.GetBounds();
|
|
LayerIntSize size = bounds.Size();
|
|
IntSize imageSize(size.width, size.height);
|
|
|
|
UpdateImageHelper helper(mImageContainer, mImageClient, imageSize);
|
|
|
|
{
|
|
RefPtr<DrawTarget> target = helper.GetDrawTarget();
|
|
if (!target) {
|
|
return false;
|
|
}
|
|
|
|
target->ClearRect(Rect(0, 0, imageSize.width, imageSize.height));
|
|
target->SetTransform(Matrix().PreTranslate(-bounds.x, -bounds.y));
|
|
RefPtr<gfxContext> ctx =
|
|
gfxContext::CreatePreservingTransformOrNull(target);
|
|
MOZ_ASSERT(ctx); // already checked the target above
|
|
|
|
Manager()->GetPaintedLayerCallback()(this,
|
|
ctx,
|
|
visibleRegion.ToUnknownRegion(), visibleRegion.ToUnknownRegion(),
|
|
DrawRegionClip::DRAW, nsIntRegion(), Manager()->GetPaintedLayerCallbackData());
|
|
|
|
if (gfxPrefs::WebRenderHighlightPaintedLayers()) {
|
|
target->SetTransform(Matrix());
|
|
target->FillRect(Rect(0, 0, imageSize.width, imageSize.height), ColorPattern(Color(1.0, 0.0, 0.0, 0.5)));
|
|
}
|
|
}
|
|
|
|
if (!helper.UpdateImage()) {
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
void
|
|
WebRenderPaintedLayer::CreateWebRenderDisplayList(wr::DisplayListBuilder& aBuilder)
|
|
{
|
|
StackingContextHelper sc(aBuilder, this);
|
|
|
|
LayerRect rect = Bounds();
|
|
DumpLayerInfo("PaintedLayer", rect);
|
|
|
|
LayerRect clipRect = ClipRect().valueOr(rect);
|
|
Maybe<WrImageMask> mask = BuildWrMaskLayer(true);
|
|
WrClipRegion clip = aBuilder.BuildClipRegion(
|
|
sc.ToRelativeWrRect(clipRect),
|
|
mask.ptrOr(nullptr));
|
|
|
|
WrImageKey key = GetImageKey();
|
|
WrBridge()->AddWebRenderParentCommand(OpAddExternalImage(mExternalImageId.value(), key));
|
|
Manager()->AddImageKeyForDiscard(key);
|
|
|
|
aBuilder.PushImage(sc.ToRelativeWrRect(rect), clip, wr::ImageRendering::Auto, key);
|
|
}
|
|
|
|
void
|
|
WebRenderPaintedLayer::RenderLayer(wr::DisplayListBuilder& aBuilder)
|
|
{
|
|
if (!SetupExternalImages()) {
|
|
return;
|
|
}
|
|
|
|
if (GetVisibleRegion().IsEmpty()) {
|
|
if (gfxPrefs::LayersDump()) {
|
|
printf_stderr("PaintedLayer %p skipping\n", this->GetLayer());
|
|
}
|
|
return;
|
|
}
|
|
|
|
nsIntRegion regionToPaint;
|
|
regionToPaint.Sub(mVisibleRegion.ToUnknownRegion(), mValidRegion);
|
|
|
|
// We have something to paint but can't. This usually happens only in
|
|
// empty transactions
|
|
if (!regionToPaint.IsEmpty() && !Manager()->GetPaintedLayerCallback()) {
|
|
Manager()->SetTransactionIncomplete();
|
|
return;
|
|
}
|
|
|
|
if (!regionToPaint.IsEmpty() && Manager()->GetPaintedLayerCallback()) {
|
|
if (!UpdateImageClient()) {
|
|
return;
|
|
}
|
|
} else {
|
|
// We have an empty transaction, just reuse the old image we had before.
|
|
MOZ_ASSERT(mExternalImageId);
|
|
MOZ_ASSERT(mImageContainer->HasCurrentImage());
|
|
MOZ_ASSERT(GetInvalidRegion().IsEmpty());
|
|
}
|
|
|
|
CreateWebRenderDisplayList(aBuilder);
|
|
}
|
|
|
|
} // namespace layers
|
|
} // namespace mozilla
|