/* -*- 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 #include "DrawTargetD2D1.h" #include "DrawTargetD2D.h" #include "FilterNodeSoftware.h" #include "GradientStopsD2D.h" #include "SourceSurfaceD2D1.h" #include "SourceSurfaceD2D.h" #include "RadialGradientEffectD2D1.h" #include "HelpersD2D.h" #include "FilterNodeD2D1.h" #include "Tools.h" using namespace std; namespace mozilla { namespace gfx { uint64_t DrawTargetD2D1::mVRAMUsageDT; uint64_t DrawTargetD2D1::mVRAMUsageSS; ID2D1Factory1* DrawTargetD2D1::mFactory = nullptr; ID2D1Factory1 *D2DFactory1() { return DrawTargetD2D1::factory(); } DrawTargetD2D1::DrawTargetD2D1() : mClipsArePushed(false) { } DrawTargetD2D1::~DrawTargetD2D1() { PopAllClips(); if (mSnapshot) { // We may hold the only reference. MarkIndependent will clear mSnapshot; // keep the snapshot object alive so it doesn't get destroyed while // MarkIndependent is running. RefPtr deathGrip = mSnapshot; // mSnapshot can be treated as independent of this DrawTarget since we know // this DrawTarget won't change again. deathGrip->MarkIndependent(); // mSnapshot will be cleared now. } if (mDC) { // The only way mDC can be null is if Init failed, but it can happen and the // destructor is the only place where we need to check for it since the // DrawTarget will destroyed right after Init fails. mDC->EndDraw(); } // Targets depending on us can break that dependency, since we're obviously not going to // be modified in the future. for (auto iter = mDependentTargets.begin(); iter != mDependentTargets.end(); iter++) { (*iter)->mDependingOnTargets.erase(this); } // Our dependencies on other targets no longer matter. for (TargetSet::iterator iter = mDependingOnTargets.begin(); iter != mDependingOnTargets.end(); iter++) { (*iter)->mDependentTargets.erase(this); } } already_AddRefed DrawTargetD2D1::Snapshot() { if (mSnapshot) { RefPtr snapshot(mSnapshot); return snapshot.forget(); } PopAllClips(); mDC->Flush(); mSnapshot = new SourceSurfaceD2D1(mBitmap, mDC, mFormat, mSize, this); RefPtr snapshot(mSnapshot); return snapshot.forget(); } void DrawTargetD2D1::Flush() { mDC->Flush(); // We no longer depend on any target. for (TargetSet::iterator iter = mDependingOnTargets.begin(); iter != mDependingOnTargets.end(); iter++) { (*iter)->mDependentTargets.erase(this); } mDependingOnTargets.clear(); } void DrawTargetD2D1::DrawSurface(SourceSurface *aSurface, const Rect &aDest, const Rect &aSource, const DrawSurfaceOptions &aSurfOptions, const DrawOptions &aOptions) { PrepareForDrawing(aOptions.mCompositionOp, ColorPattern(Color())); D2D1_RECT_F samplingBounds; if (aSurfOptions.mSamplingBounds == SamplingBounds::BOUNDED) { samplingBounds = D2DRect(aSource); } else { samplingBounds = D2D1::RectF(0, 0, Float(aSurface->GetSize().width), Float(aSurface->GetSize().height)); } Float xScale = aDest.width / aSource.width; Float yScale = aDest.height / aSource.height; RefPtr brush; // Here we scale the source pattern up to the size and position where we want // it to be. Matrix transform; transform.PreTranslate(aDest.x - aSource.x * xScale, aDest.y - aSource.y * yScale); transform.PreScale(xScale, yScale); RefPtr image = GetImageForSurface(aSurface, transform, ExtendMode::CLAMP); if (!image) { gfxWarning() << *this << ": Unable to get D2D image for surface."; return; } RefPtr bitmap; if (aSurface->GetType() == SurfaceType::D2D1_1_IMAGE) { // If this is called with a DataSourceSurface it might do a partial upload // that our DrawBitmap call doesn't support. image->QueryInterface((ID2D1Bitmap**)byRef(bitmap)); } if (bitmap && aSurfOptions.mSamplingBounds == SamplingBounds::UNBOUNDED) { mDC->DrawBitmap(bitmap, D2DRect(aDest), aOptions.mAlpha, D2DFilter(aSurfOptions.mFilter), D2DRect(aSource)); } else { // This has issues ignoring the alpha channel on windows 7 with images marked opaque. MOZ_ASSERT(aSurface->GetFormat() != SurfaceFormat::B8G8R8X8); mDC->CreateImageBrush(image, D2D1::ImageBrushProperties(samplingBounds, D2D1_EXTEND_MODE_CLAMP, D2D1_EXTEND_MODE_CLAMP, D2DInterpolationMode(aSurfOptions.mFilter)), D2D1::BrushProperties(aOptions.mAlpha, D2DMatrix(transform)), byRef(brush)); mDC->FillRectangle(D2DRect(aDest), brush); } FinalizeDrawing(aOptions.mCompositionOp, ColorPattern(Color())); } void DrawTargetD2D1::DrawFilter(FilterNode *aNode, const Rect &aSourceRect, const Point &aDestPoint, const DrawOptions &aOptions) { if (aNode->GetBackendType() != FILTER_BACKEND_DIRECT2D1_1) { gfxWarning() << *this << ": Incompatible filter passed to DrawFilter."; return; } PrepareForDrawing(aOptions.mCompositionOp, ColorPattern(Color())); mDC->SetAntialiasMode(D2DAAMode(aOptions.mAntialiasMode)); FilterNodeD2D1* node = static_cast(aNode); node->WillDraw(this); mDC->DrawImage(node->OutputEffect(), D2DPoint(aDestPoint), D2DRect(aSourceRect)); FinalizeDrawing(aOptions.mCompositionOp, ColorPattern(Color())); } void DrawTargetD2D1::DrawSurfaceWithShadow(SourceSurface *aSurface, const Point &aDest, const Color &aColor, const Point &aOffset, Float aSigma, CompositionOp aOperator) { MarkChanged(); mDC->SetTransform(D2D1::IdentityMatrix()); mTransformDirty = true; Matrix mat; RefPtr image = GetImageForSurface(aSurface, mat, ExtendMode::CLAMP); if (!image) { gfxWarning() << "Couldn't get image for surface."; return; } if (!mat.IsIdentity()) { gfxDebug() << *this << ": At this point complex partial uploads are not supported for Shadow surfaces."; return; } // Step 1, create the shadow effect. RefPtr shadowEffect; HRESULT hr = mDC->CreateEffect(CLSID_D2D1Shadow, byRef(shadowEffect)); if (FAILED(hr) || !shadowEffect) { gfxWarning() << "Failed to create shadow effect. Code: " << hexa(hr); return; } shadowEffect->SetInput(0, image); shadowEffect->SetValue(D2D1_SHADOW_PROP_BLUR_STANDARD_DEVIATION, aSigma); D2D1_VECTOR_4F color = { aColor.r, aColor.g, aColor.b, aColor.a }; shadowEffect->SetValue(D2D1_SHADOW_PROP_COLOR, color); D2D1_POINT_2F shadowPoint = D2DPoint(aDest + aOffset); mDC->DrawImage(shadowEffect, &shadowPoint, nullptr, D2D1_INTERPOLATION_MODE_LINEAR, D2DCompositionMode(aOperator)); D2D1_POINT_2F imgPoint = D2DPoint(aDest); mDC->DrawImage(image, &imgPoint, nullptr, D2D1_INTERPOLATION_MODE_LINEAR, D2DCompositionMode(aOperator)); } void DrawTargetD2D1::ClearRect(const Rect &aRect) { MarkChanged(); PopAllClips(); PushClipRect(aRect); if (mTransformDirty || !mTransform.IsIdentity()) { mDC->SetTransform(D2D1::IdentityMatrix()); mTransformDirty = true; } D2D1_RECT_F clipRect; bool isPixelAligned; if (mTransform.IsRectilinear() && GetDeviceSpaceClipRect(clipRect, isPixelAligned)) { mDC->PushAxisAlignedClip(clipRect, isPixelAligned ? D2D1_ANTIALIAS_MODE_ALIASED : D2D1_ANTIALIAS_MODE_PER_PRIMITIVE); mDC->Clear(); mDC->PopAxisAlignedClip(); PopClip(); return; } mDC->SetTarget(mTempBitmap); mDC->Clear(); IntRect addClipRect; RefPtr geom = GetClippedGeometry(&addClipRect); RefPtr brush; mDC->CreateSolidColorBrush(D2D1::ColorF(D2D1::ColorF::White), byRef(brush)); mDC->PushAxisAlignedClip(D2D1::RectF(addClipRect.x, addClipRect.y, addClipRect.XMost(), addClipRect.YMost()), D2D1_ANTIALIAS_MODE_PER_PRIMITIVE); mDC->FillGeometry(geom, brush); mDC->PopAxisAlignedClip(); mDC->SetTarget(mBitmap); mDC->DrawImage(mTempBitmap, D2D1_INTERPOLATION_MODE_NEAREST_NEIGHBOR, D2D1_COMPOSITE_MODE_DESTINATION_OUT); PopClip(); return; } void DrawTargetD2D1::MaskSurface(const Pattern &aSource, SourceSurface *aMask, Point aOffset, const DrawOptions &aOptions) { MarkChanged(); RefPtr bitmap; RefPtr image = GetImageForSurface(aMask, ExtendMode::CLAMP); if (!image) { gfxWarning() << "Failed to get image for surface."; return; } PrepareForDrawing(aOptions.mCompositionOp, aSource); // FillOpacityMask only works if the antialias mode is MODE_ALIASED mDC->SetAntialiasMode(D2D1_ANTIALIAS_MODE_ALIASED); IntSize size = aMask->GetSize(); Rect maskRect = Rect(0.f, 0.f, Float(size.width), Float(size.height)); image->QueryInterface((ID2D1Bitmap**)&bitmap); if (!bitmap) { gfxWarning() << "FillOpacityMask only works with Bitmap source surfaces."; return; } Rect dest = Rect(aOffset.x, aOffset.y, Float(size.width), Float(size.height)); RefPtr brush = CreateBrushForPattern(aSource, aOptions.mAlpha); mDC->FillOpacityMask(bitmap, brush, D2D1_OPACITY_MASK_CONTENT_GRAPHICS, D2DRect(dest), D2DRect(maskRect)); mDC->SetAntialiasMode(D2D1_ANTIALIAS_MODE_PER_PRIMITIVE); FinalizeDrawing(aOptions.mCompositionOp, aSource); } void DrawTargetD2D1::CopySurface(SourceSurface *aSurface, const IntRect &aSourceRect, const IntPoint &aDestination) { MarkChanged(); PopAllClips(); mDC->SetTransform(D2D1::IdentityMatrix()); mTransformDirty = true; Matrix mat; RefPtr image = GetImageForSurface(aSurface, mat, ExtendMode::CLAMP); if (!image) { gfxWarning() << "Couldn't get image for surface."; return; } if (!mat.IsIdentity()) { gfxDebug() << *this << ": At this point complex partial uploads are not supported for CopySurface."; return; } RefPtr bitmap; image->QueryInterface((ID2D1Bitmap**)byRef(bitmap)); if (bitmap && mFormat == SurfaceFormat::A8) { RefPtr brush; mDC->CreateSolidColorBrush(D2D1::ColorF(D2D1::ColorF::White), D2D1::BrushProperties(), byRef(brush)); mDC->SetAntialiasMode(D2D1_ANTIALIAS_MODE_ALIASED); mDC->SetPrimitiveBlend(D2D1_PRIMITIVE_BLEND_COPY); mDC->FillOpacityMask(bitmap, brush, D2D1_OPACITY_MASK_CONTENT_GRAPHICS); mDC->SetAntialiasMode(D2D1_ANTIALIAS_MODE_PER_PRIMITIVE); mDC->SetPrimitiveBlend(D2D1_PRIMITIVE_BLEND_SOURCE_OVER); return; } Rect srcRect(Float(aSourceRect.x), Float(aSourceRect.y), Float(aSourceRect.width), Float(aSourceRect.height)); Rect dstRect(Float(aDestination.x), Float(aDestination.y), Float(aSourceRect.width), Float(aSourceRect.height)); if (bitmap) { mDC->SetPrimitiveBlend(D2D1_PRIMITIVE_BLEND_COPY); mDC->DrawBitmap(bitmap, D2DRect(dstRect), 1.0f, D2D1_BITMAP_INTERPOLATION_MODE_NEAREST_NEIGHBOR, D2DRect(srcRect)); mDC->SetPrimitiveBlend(D2D1_PRIMITIVE_BLEND_SOURCE_OVER); return; } mDC->DrawImage(image, D2D1::Point2F(Float(aDestination.x), Float(aDestination.y)), D2DRect(srcRect), D2D1_INTERPOLATION_MODE_NEAREST_NEIGHBOR, D2D1_COMPOSITE_MODE_BOUNDED_SOURCE_COPY); } void DrawTargetD2D1::FillRect(const Rect &aRect, const Pattern &aPattern, const DrawOptions &aOptions) { PrepareForDrawing(aOptions.mCompositionOp, aPattern); mDC->SetAntialiasMode(D2DAAMode(aOptions.mAntialiasMode)); RefPtr brush = CreateBrushForPattern(aPattern, aOptions.mAlpha); mDC->FillRectangle(D2DRect(aRect), brush); FinalizeDrawing(aOptions.mCompositionOp, aPattern); } void DrawTargetD2D1::StrokeRect(const Rect &aRect, const Pattern &aPattern, const StrokeOptions &aStrokeOptions, const DrawOptions &aOptions) { PrepareForDrawing(aOptions.mCompositionOp, aPattern); mDC->SetAntialiasMode(D2DAAMode(aOptions.mAntialiasMode)); RefPtr brush = CreateBrushForPattern(aPattern, aOptions.mAlpha); RefPtr strokeStyle = CreateStrokeStyleForOptions(aStrokeOptions); mDC->DrawRectangle(D2DRect(aRect), brush, aStrokeOptions.mLineWidth, strokeStyle); FinalizeDrawing(aOptions.mCompositionOp, aPattern); } void DrawTargetD2D1::StrokeLine(const Point &aStart, const Point &aEnd, const Pattern &aPattern, const StrokeOptions &aStrokeOptions, const DrawOptions &aOptions) { PrepareForDrawing(aOptions.mCompositionOp, aPattern); mDC->SetAntialiasMode(D2DAAMode(aOptions.mAntialiasMode)); RefPtr brush = CreateBrushForPattern(aPattern, aOptions.mAlpha); RefPtr strokeStyle = CreateStrokeStyleForOptions(aStrokeOptions); mDC->DrawLine(D2DPoint(aStart), D2DPoint(aEnd), brush, aStrokeOptions.mLineWidth, strokeStyle); FinalizeDrawing(aOptions.mCompositionOp, aPattern); } void DrawTargetD2D1::Stroke(const Path *aPath, const Pattern &aPattern, const StrokeOptions &aStrokeOptions, const DrawOptions &aOptions) { if (aPath->GetBackendType() != BackendType::DIRECT2D1_1) { gfxDebug() << *this << ": Ignoring drawing call for incompatible path."; return; } const PathD2D *d2dPath = static_cast(aPath); PrepareForDrawing(aOptions.mCompositionOp, aPattern); mDC->SetAntialiasMode(D2DAAMode(aOptions.mAntialiasMode)); RefPtr brush = CreateBrushForPattern(aPattern, aOptions.mAlpha); RefPtr strokeStyle = CreateStrokeStyleForOptions(aStrokeOptions); mDC->DrawGeometry(d2dPath->mGeometry, brush, aStrokeOptions.mLineWidth, strokeStyle); FinalizeDrawing(aOptions.mCompositionOp, aPattern); } void DrawTargetD2D1::Fill(const Path *aPath, const Pattern &aPattern, const DrawOptions &aOptions) { if (aPath->GetBackendType() != BackendType::DIRECT2D1_1) { gfxDebug() << *this << ": Ignoring drawing call for incompatible path."; return; } const PathD2D *d2dPath = static_cast(aPath); PrepareForDrawing(aOptions.mCompositionOp, aPattern); mDC->SetAntialiasMode(D2DAAMode(aOptions.mAntialiasMode)); RefPtr brush = CreateBrushForPattern(aPattern, aOptions.mAlpha); mDC->FillGeometry(d2dPath->mGeometry, brush); FinalizeDrawing(aOptions.mCompositionOp, aPattern); } void DrawTargetD2D1::FillGlyphs(ScaledFont *aFont, const GlyphBuffer &aBuffer, const Pattern &aPattern, const DrawOptions &aOptions, const GlyphRenderingOptions *aRenderingOptions) { if (aFont->GetType() != FontType::DWRITE) { gfxDebug() << *this << ": Ignoring drawing call for incompatible font."; return; } ScaledFontDWrite *font = static_cast(aFont); IDWriteRenderingParams *params = nullptr; if (aRenderingOptions) { if (aRenderingOptions->GetType() != FontType::DWRITE) { gfxDebug() << *this << ": Ignoring incompatible GlyphRenderingOptions."; // This should never happen. MOZ_ASSERT(false); } else { params = static_cast(aRenderingOptions)->mParams; } } AntialiasMode aaMode = font->GetDefaultAAMode(); if (aOptions.mAntialiasMode != AntialiasMode::DEFAULT) { aaMode = aOptions.mAntialiasMode; } PrepareForDrawing(aOptions.mCompositionOp, aPattern); bool forceClearType = false; if (mFormat == SurfaceFormat::B8G8R8A8 && mPermitSubpixelAA && aOptions.mCompositionOp == CompositionOp::OP_OVER && aaMode == AntialiasMode::SUBPIXEL) { forceClearType = true; } D2D1_TEXT_ANTIALIAS_MODE d2dAAMode = D2D1_TEXT_ANTIALIAS_MODE_DEFAULT; switch (aaMode) { case AntialiasMode::NONE: d2dAAMode = D2D1_TEXT_ANTIALIAS_MODE_ALIASED; break; case AntialiasMode::GRAY: d2dAAMode = D2D1_TEXT_ANTIALIAS_MODE_GRAYSCALE; break; case AntialiasMode::SUBPIXEL: d2dAAMode = D2D1_TEXT_ANTIALIAS_MODE_CLEARTYPE; break; default: d2dAAMode = D2D1_TEXT_ANTIALIAS_MODE_DEFAULT; } if (d2dAAMode == D2D1_TEXT_ANTIALIAS_MODE_CLEARTYPE && mFormat != SurfaceFormat::B8G8R8X8 && !forceClearType) { d2dAAMode = D2D1_TEXT_ANTIALIAS_MODE_GRAYSCALE; } mDC->SetTextAntialiasMode(d2dAAMode); if (params != mTextRenderingParams) { mDC->SetTextRenderingParams(params); mTextRenderingParams = params; } RefPtr brush = CreateBrushForPattern(aPattern, aOptions.mAlpha); AutoDWriteGlyphRun autoRun; DWriteGlyphRunFromGlyphs(aBuffer, font, &autoRun); bool needsRepushedLayers = false; if (d2dAAMode == D2D1_TEXT_ANTIALIAS_MODE_CLEARTYPE && mFormat != SurfaceFormat::B8G8R8X8) { D2D1_RECT_F rect; bool isAligned; needsRepushedLayers = mPushedClips.size() && !GetDeviceSpaceClipRect(rect, isAligned); // If we have a complex clip in our stack and we have a transparent // background, and subpixel AA is permitted, we need to repush our layer // stack limited by the glyph run bounds initializing our layers for // subpixel AA. if (needsRepushedLayers) { mDC->GetGlyphRunWorldBounds(D2D1::Point2F(), &autoRun, DWRITE_MEASURING_MODE_NATURAL, &rect); rect.left = std::floor(rect.left); rect.right = std::ceil(rect.right); rect.top = std::floor(rect.top); rect.bottom = std::ceil(rect.bottom); PopAllClips(); if (!mTransform.IsRectilinear()) { // We must limit the pixels we touch to the -user space- bounds of // the glyphs being drawn. In order not to get transparent pixels // copied up in our pushed layer stack. D2D1_RECT_F userRect; mDC->SetTransform(D2D1::IdentityMatrix()); mDC->GetGlyphRunWorldBounds(D2D1::Point2F(), &autoRun, DWRITE_MEASURING_MODE_NATURAL, &userRect); RefPtr path; D2DFactory()->CreatePathGeometry(byRef(path)); RefPtr sink; path->Open(byRef(sink)); sink->BeginFigure(D2D1::Point2F(userRect.left, userRect.top), D2D1_FIGURE_BEGIN_FILLED); sink->AddLine(D2D1::Point2F(userRect.right, userRect.top)); sink->AddLine(D2D1::Point2F(userRect.right, userRect.bottom)); sink->AddLine(D2D1::Point2F(userRect.left, userRect.bottom)); sink->EndFigure(D2D1_FIGURE_END_CLOSED); sink->Close(); mDC->PushLayer(D2D1::LayerParameters1(D2D1::InfiniteRect(), path, D2D1_ANTIALIAS_MODE_ALIASED, D2DMatrix(mTransform), 1.0f, nullptr, D2D1_LAYER_OPTIONS1_INITIALIZE_FROM_BACKGROUND | D2D1_LAYER_OPTIONS1_IGNORE_ALPHA), nullptr); } PushClipsToDC(mDC, true, rect); mDC->SetTransform(D2DMatrix(mTransform)); } } if (brush) { mDC->DrawGlyphRun(D2D1::Point2F(), &autoRun, brush); } if (needsRepushedLayers) { PopClipsFromDC(mDC); if (!mTransform.IsRectilinear()) { mDC->PopLayer(); } } FinalizeDrawing(aOptions.mCompositionOp, aPattern); } void DrawTargetD2D1::Mask(const Pattern &aSource, const Pattern &aMask, const DrawOptions &aOptions) { PrepareForDrawing(aOptions.mCompositionOp, aSource); RefPtr source = CreateBrushForPattern(aSource, aOptions.mAlpha); RefPtr mask = CreateBrushForPattern(aMask, 1.0f); mDC->PushLayer(D2D1::LayerParameters(D2D1::InfiniteRect(), nullptr, D2D1_ANTIALIAS_MODE_PER_PRIMITIVE, D2D1::IdentityMatrix(), 1.0f, mask), nullptr); Rect rect(0, 0, (Float)mSize.width, (Float)mSize.height); Matrix mat = mTransform; mat.Invert(); mDC->FillRectangle(D2DRect(mat.TransformBounds(rect)), source); mDC->PopLayer(); FinalizeDrawing(aOptions.mCompositionOp, aSource); } void DrawTargetD2D1::PushClip(const Path *aPath) { if (aPath->GetBackendType() != BackendType::DIRECT2D1_1) { gfxDebug() << *this << ": Ignoring clipping call for incompatible path."; return; } mCurrentClippedGeometry = nullptr; RefPtr pathD2D = static_cast(const_cast(aPath)); PushedClip clip; clip.mTransform = D2DMatrix(mTransform); clip.mPath = pathD2D; pathD2D->mGeometry->GetBounds(clip.mTransform, &clip.mBounds); mPushedClips.push_back(clip); // The transform of clips is relative to the world matrix, since we use the total // transform for the clips, make the world matrix identity. mDC->SetTransform(D2D1::IdentityMatrix()); mTransformDirty = true; if (mClipsArePushed) { PushD2DLayer(mDC, pathD2D->mGeometry, clip.mTransform); } } void DrawTargetD2D1::PushClipRect(const Rect &aRect) { if (!mTransform.IsRectilinear()) { // Whoops, this isn't a rectangle in device space, Direct2D will not deal // with this transform the way we want it to. // See remarks: http://msdn.microsoft.com/en-us/library/dd316860%28VS.85%29.aspx RefPtr pathBuilder = CreatePathBuilder(); pathBuilder->MoveTo(aRect.TopLeft()); pathBuilder->LineTo(aRect.TopRight()); pathBuilder->LineTo(aRect.BottomRight()); pathBuilder->LineTo(aRect.BottomLeft()); pathBuilder->Close(); RefPtr path = pathBuilder->Finish(); return PushClip(path); } mCurrentClippedGeometry = nullptr; PushedClip clip; Rect rect = mTransform.TransformBounds(aRect); IntRect intRect; clip.mIsPixelAligned = rect.ToIntRect(&intRect); // Do not store the transform, just store the device space rectangle directly. clip.mBounds = D2DRect(rect); mPushedClips.push_back(clip); mDC->SetTransform(D2D1::IdentityMatrix()); mTransformDirty = true; if (mClipsArePushed) { mDC->PushAxisAlignedClip(clip.mBounds, clip.mIsPixelAligned ? D2D1_ANTIALIAS_MODE_ALIASED : D2D1_ANTIALIAS_MODE_PER_PRIMITIVE); } } void DrawTargetD2D1::PopClip() { mCurrentClippedGeometry = nullptr; if (mClipsArePushed) { if (mPushedClips.back().mPath) { mDC->PopLayer(); } else { mDC->PopAxisAlignedClip(); } } mPushedClips.pop_back(); } already_AddRefed DrawTargetD2D1::CreateSourceSurfaceFromData(unsigned char *aData, const IntSize &aSize, int32_t aStride, SurfaceFormat aFormat) const { RefPtr bitmap; HRESULT hr = mDC->CreateBitmap(D2DIntSize(aSize), aData, aStride, D2D1::BitmapProperties1(D2D1_BITMAP_OPTIONS_NONE, D2DPixelFormat(aFormat)), byRef(bitmap)); if (FAILED(hr) || !bitmap) { gfxCriticalError(CriticalLog::DefaultOptions(Factory::ReasonableSurfaceSize(aSize))) << "[D2D1.1] 1CreateBitmap failure " << aSize << " Code: " << hexa(hr) << " format " << (int)aFormat; return nullptr; } return MakeAndAddRef(bitmap.get(), mDC, aFormat, aSize); } already_AddRefed DrawTargetD2D1::CreateSimilarDrawTarget(const IntSize &aSize, SurfaceFormat aFormat) const { RefPtr dt = new DrawTargetD2D1(); if (!dt->Init(aSize, aFormat)) { return nullptr; } return dt.forget(); } already_AddRefed DrawTargetD2D1::CreatePathBuilder(FillRule aFillRule) const { RefPtr path; HRESULT hr = factory()->CreatePathGeometry(byRef(path)); if (FAILED(hr)) { gfxWarning() << *this << ": Failed to create Direct2D Path Geometry. Code: " << hexa(hr); return nullptr; } RefPtr sink; hr = path->Open(byRef(sink)); if (FAILED(hr)) { gfxWarning() << *this << ": Failed to access Direct2D Path Geometry. Code: " << hexa(hr); return nullptr; } if (aFillRule == FillRule::FILL_WINDING) { sink->SetFillMode(D2D1_FILL_MODE_WINDING); } return MakeAndAddRef(sink, path, aFillRule, BackendType::DIRECT2D1_1); } already_AddRefed DrawTargetD2D1::CreateGradientStops(GradientStop *rawStops, uint32_t aNumStops, ExtendMode aExtendMode) const { if (aNumStops == 0) { gfxWarning() << *this << ": Failed to create GradientStopCollection with no stops."; return nullptr; } D2D1_GRADIENT_STOP *stops = new D2D1_GRADIENT_STOP[aNumStops]; for (uint32_t i = 0; i < aNumStops; i++) { stops[i].position = rawStops[i].offset; stops[i].color = D2DColor(rawStops[i].color); } RefPtr stopCollection; HRESULT hr = mDC->CreateGradientStopCollection(stops, aNumStops, D2D1_GAMMA_2_2, D2DExtend(aExtendMode), byRef(stopCollection)); delete [] stops; if (FAILED(hr)) { gfxWarning() << *this << ": Failed to create GradientStopCollection. Code: " << hexa(hr); return nullptr; } return MakeAndAddRef(stopCollection, Factory::GetDirect3D11Device()); } already_AddRefed DrawTargetD2D1::CreateFilter(FilterType aType) { return FilterNodeD2D1::Create(mDC, aType); } bool DrawTargetD2D1::Init(ID3D11Texture2D* aTexture, SurfaceFormat aFormat) { HRESULT hr; ID2D1Device* device = Factory::GetD2D1Device(); if (!device) { return false; } hr = device->CreateDeviceContext(D2D1_DEVICE_CONTEXT_OPTIONS_ENABLE_MULTITHREADED_OPTIMIZATIONS, byRef(mDC)); if (FAILED(hr)) { gfxCriticalError() <<"[D2D1.1] 1Failed to create a DeviceContext, code: " << hexa(hr) << " format " << (int)aFormat; return false; } RefPtr dxgiSurface; aTexture->QueryInterface(__uuidof(IDXGISurface), (void**)((IDXGISurface**)byRef(dxgiSurface))); if (!dxgiSurface) { gfxCriticalError() <<"[D2D1.1] Failed to obtain a DXGI surface."; return false; } D2D1_BITMAP_PROPERTIES1 props; props.dpiX = 96; props.dpiY = 96; props.pixelFormat = D2DPixelFormat(aFormat); props.colorContext = nullptr; props.bitmapOptions = D2D1_BITMAP_OPTIONS_TARGET; hr = mDC->CreateBitmapFromDxgiSurface(dxgiSurface, props, (ID2D1Bitmap1**)byRef(mBitmap)); if (FAILED(hr)) { gfxCriticalError() << "[D2D1.1] CreateBitmapFromDxgiSurface failure Code: " << hexa(hr) << " format " << (int)aFormat; return false; } mFormat = aFormat; D3D11_TEXTURE2D_DESC desc; aTexture->GetDesc(&desc); desc.Format = DXGI_FORMAT_B8G8R8A8_UNORM; mSize.width = desc.Width; mSize.height = desc.Height; props.pixelFormat.alphaMode = D2D1_ALPHA_MODE_PREMULTIPLIED; props.pixelFormat.format = DXGI_FORMAT_B8G8R8A8_UNORM; hr = mDC->CreateBitmap(D2DIntSize(mSize), nullptr, 0, props, (ID2D1Bitmap1**)byRef(mTempBitmap)); if (FAILED(hr)) { gfxCriticalError(CriticalLog::DefaultOptions(Factory::ReasonableSurfaceSize(mSize))) << "[D2D1.1] 2CreateBitmap failure " << mSize << " Code: " << hexa(hr); return false; } // This single solid color brush system is not very 'threadsafe', however, // issueing multiple drawing commands simultaneously to a single drawtarget // from multiple threads is unexpected since there's no way to guarantee // ordering in that situation anyway. hr = mDC->CreateSolidColorBrush(D2D1::ColorF(0, 0), byRef(mSolidColorBrush)); if (FAILED(hr)) { gfxCriticalError() << "[D2D1.1] Failure creating solid color brush."; return false; } mDC->SetTarget(mBitmap); mDC->BeginDraw(); return true; } bool DrawTargetD2D1::Init(const IntSize &aSize, SurfaceFormat aFormat) { HRESULT hr; ID2D1Device* device = Factory::GetD2D1Device(); if (!device) { return false; } hr = device->CreateDeviceContext(D2D1_DEVICE_CONTEXT_OPTIONS_ENABLE_MULTITHREADED_OPTIMIZATIONS, byRef(mDC)); if (FAILED(hr)) { gfxCriticalError() <<"[D2D1.1] 2Failed to create a DeviceContext, code: " << hexa(hr) << " format " << (int)aFormat; return false; } if (mDC->GetMaximumBitmapSize() < UINT32(aSize.width) || mDC->GetMaximumBitmapSize() < UINT32(aSize.height)) { // This is 'ok', so don't assert gfxCriticalError(CriticalLog::DefaultOptions(false)) << "[D2D1.1] Attempt to use unsupported surface size " << aSize; return false; } D2D1_BITMAP_PROPERTIES1 props; props.dpiX = 96; props.dpiY = 96; props.pixelFormat = D2DPixelFormat(aFormat); props.colorContext = nullptr; props.bitmapOptions = D2D1_BITMAP_OPTIONS_TARGET; hr = mDC->CreateBitmap(D2DIntSize(aSize), nullptr, 0, props, (ID2D1Bitmap1**)byRef(mBitmap)); if (FAILED(hr)) { gfxCriticalError() << "[D2D1.1] 3CreateBitmap failure " << aSize << " Code: " << hexa(hr) << " format " << (int)aFormat; return false; } props.pixelFormat.alphaMode = D2D1_ALPHA_MODE_PREMULTIPLIED; props.pixelFormat.format = DXGI_FORMAT_B8G8R8A8_UNORM; hr = mDC->CreateBitmap(D2DIntSize(aSize), nullptr, 0, props, (ID2D1Bitmap1**)byRef(mTempBitmap)); if (FAILED(hr)) { gfxCriticalError(CriticalLog::DefaultOptions(Factory::ReasonableSurfaceSize(aSize))) << "[D2D1.1] failed to create new TempBitmap " << aSize << " Code: " << hexa(hr); return false; } mDC->SetTarget(mBitmap); hr = mDC->CreateSolidColorBrush(D2D1::ColorF(0, 0), byRef(mSolidColorBrush)); if (FAILED(hr)) { gfxCriticalError() << "[D2D1.1] Failure creating solid color brush."; return false; } mDC->BeginDraw(); mDC->Clear(); mFormat = aFormat; mSize = aSize; return true; } /** * Private helpers. */ uint32_t DrawTargetD2D1::GetByteSize() const { return mSize.width * mSize.height * BytesPerPixel(mFormat); } ID2D1Factory1* DrawTargetD2D1::factory() { if (mFactory) { return mFactory; } ID2D1Factory* d2dFactory = D2DFactory(); if (!d2dFactory) { return nullptr; } HRESULT hr = d2dFactory->QueryInterface((ID2D1Factory1**)&mFactory); if (FAILED(hr)) { return nullptr; } RadialGradientEffectD2D1::Register(mFactory); return mFactory; } void DrawTargetD2D1::CleanupD2D() { if (mFactory) { RadialGradientEffectD2D1::Unregister(mFactory); mFactory->Release(); mFactory = nullptr; } } void DrawTargetD2D1::MarkChanged() { if (mSnapshot) { if (mSnapshot->hasOneRef()) { // Just destroy it, since no-one else knows about it. mSnapshot = nullptr; } else { mSnapshot->DrawTargetWillChange(); // The snapshot will no longer depend on this target. MOZ_ASSERT(!mSnapshot); } } if (mDependentTargets.size()) { // Copy mDependentTargets since the Flush()es below will modify it. TargetSet tmpTargets = mDependentTargets; for (TargetSet::iterator iter = tmpTargets.begin(); iter != tmpTargets.end(); iter++) { (*iter)->Flush(); } // The Flush() should have broken all dependencies on this target. MOZ_ASSERT(!mDependentTargets.size()); } } void DrawTargetD2D1::PrepareForDrawing(CompositionOp aOp, const Pattern &aPattern) { MarkChanged(); if (D2DSupportsPrimitiveBlendMode(aOp) && IsPatternSupportedByD2D(aPattern)) { // It's important to do this before FlushTransformToDC! As this will cause // the transform to become dirty. PushAllClips(); FlushTransformToDC(); if (aOp != CompositionOp::OP_OVER) mDC->SetPrimitiveBlend(D2DPrimitiveBlendMode(aOp)); return; } PopAllClips(); mDC->SetTarget(mTempBitmap); mDC->Clear(D2D1::ColorF(0, 0)); PushAllClips(); FlushTransformToDC(); } void DrawTargetD2D1::FinalizeDrawing(CompositionOp aOp, const Pattern &aPattern) { bool patternSupported = IsPatternSupportedByD2D(aPattern); if (D2DSupportsPrimitiveBlendMode(aOp) && patternSupported) { if (aOp != CompositionOp::OP_OVER) mDC->SetPrimitiveBlend(D2D1_PRIMITIVE_BLEND_SOURCE_OVER); return; } PopAllClips(); RefPtr image; mDC->GetTarget(byRef(image)); mDC->SetTarget(mBitmap); mDC->SetTransform(D2D1::IdentityMatrix()); mTransformDirty = true; if (patternSupported) { if (D2DSupportsCompositeMode(aOp)) { D2D1_RECT_F rect; bool isAligned; RefPtr tmpBitmap; bool clipIsComplex = mPushedClips.size() && !GetDeviceSpaceClipRect(rect, isAligned); if (clipIsComplex) { if (!IsOperatorBoundByMask(aOp)) { HRESULT hr = mDC->CreateBitmap(D2DIntSize(mSize), D2D1::BitmapProperties(D2DPixelFormat(mFormat)), byRef(tmpBitmap)); if (FAILED(hr)) { gfxCriticalError(CriticalLog::DefaultOptions(Factory::ReasonableSurfaceSize(mSize))) << "[D2D1.1] 6CreateBitmap failure " << mSize << " Code: " << hexa(hr) << " format " << (int)mFormat; // For now, crash in this scenario; this should happen because tmpBitmap is // null and CopyFromBitmap call below dereferences it. // return; } mDC->Flush(); tmpBitmap->CopyFromBitmap(nullptr, mBitmap, nullptr); } } else { PushAllClips(); } mDC->DrawImage(image, D2D1_INTERPOLATION_MODE_NEAREST_NEIGHBOR, D2DCompositionMode(aOp)); if (tmpBitmap) { RefPtr brush; RefPtr inverseGeom = GetInverseClippedGeometry(); mDC->CreateBitmapBrush(tmpBitmap, byRef(brush)); mDC->SetPrimitiveBlend(D2D1_PRIMITIVE_BLEND_COPY); mDC->FillGeometry(inverseGeom, brush); mDC->SetPrimitiveBlend(D2D1_PRIMITIVE_BLEND_SOURCE_OVER); } return; } if (!mBlendEffect) { HRESULT hr = mDC->CreateEffect(CLSID_D2D1Blend, byRef(mBlendEffect)); if (FAILED(hr) || !mBlendEffect) { gfxWarning() << "Failed to create blend effect!"; return; } } RefPtr tmpBitmap; HRESULT hr = mDC->CreateBitmap(D2DIntSize(mSize), D2D1::BitmapProperties(D2DPixelFormat(mFormat)), byRef(tmpBitmap)); if (FAILED(hr)) { gfxCriticalError(CriticalLog::DefaultOptions(Factory::ReasonableSurfaceSize(mSize))) << "[D2D1.1] 5CreateBitmap failure " << mSize << " Code: " << hexa(hr) << " format " << (int)mFormat; return; } // This flush is important since the copy method will not know about the context drawing to the surface. // We also need to pop all the clips to make sure any drawn content will have made it to the final bitmap. mDC->Flush(); // We need to use a copy here because affects don't accept a surface on // both their in- and outputs. tmpBitmap->CopyFromBitmap(nullptr, mBitmap, nullptr); mBlendEffect->SetInput(0, tmpBitmap); mBlendEffect->SetInput(1, mTempBitmap); mBlendEffect->SetValue(D2D1_BLEND_PROP_MODE, D2DBlendMode(aOp)); PushAllClips(); mDC->DrawImage(mBlendEffect, D2D1_INTERPOLATION_MODE_NEAREST_NEIGHBOR, D2D1_COMPOSITE_MODE_BOUNDED_SOURCE_COPY); return; } const RadialGradientPattern *pat = static_cast(&aPattern); if (pat->mCenter1 == pat->mCenter2 && pat->mRadius1 == pat->mRadius2) { // Draw nothing! return; } PushAllClips(); RefPtr radialGradientEffect; HRESULT hr = mDC->CreateEffect(CLSID_RadialGradientEffect, byRef(radialGradientEffect)); if (FAILED(hr) || !radialGradientEffect) { gfxWarning() << "Failed to create radial gradient effect. Code: " << hexa(hr); return; } radialGradientEffect->SetValue(RADIAL_PROP_STOP_COLLECTION, static_cast(pat->mStops.get())->mStopCollection); radialGradientEffect->SetValue(RADIAL_PROP_CENTER_1, D2D1::Vector2F(pat->mCenter1.x, pat->mCenter1.y)); radialGradientEffect->SetValue(RADIAL_PROP_CENTER_2, D2D1::Vector2F(pat->mCenter2.x, pat->mCenter2.y)); radialGradientEffect->SetValue(RADIAL_PROP_RADIUS_1, pat->mRadius1); radialGradientEffect->SetValue(RADIAL_PROP_RADIUS_2, pat->mRadius2); radialGradientEffect->SetValue(RADIAL_PROP_RADIUS_2, pat->mRadius2); radialGradientEffect->SetValue(RADIAL_PROP_TRANSFORM, D2DMatrix(pat->mMatrix * mTransform)); radialGradientEffect->SetInput(0, image); mDC->DrawImage(radialGradientEffect, D2D1_INTERPOLATION_MODE_NEAREST_NEIGHBOR, D2DCompositionMode(aOp)); } void DrawTargetD2D1::AddDependencyOnSource(SourceSurfaceD2D1* aSource) { if (aSource->mDrawTarget && !mDependingOnTargets.count(aSource->mDrawTarget)) { aSource->mDrawTarget->mDependentTargets.insert(this); mDependingOnTargets.insert(aSource->mDrawTarget); } } static D2D1_RECT_F IntersectRect(const D2D1_RECT_F& aRect1, const D2D1_RECT_F& aRect2) { D2D1_RECT_F result; result.left = max(aRect1.left, aRect2.left); result.top = max(aRect1.top, aRect2.top); result.right = min(aRect1.right, aRect2.right); result.bottom = min(aRect1.bottom, aRect2.bottom); result.right = max(result.right, result.left); result.bottom = max(result.bottom, result.top); return result; } bool DrawTargetD2D1::GetDeviceSpaceClipRect(D2D1_RECT_F& aClipRect, bool& aIsPixelAligned) { if (!mPushedClips.size()) { return false; } aClipRect = D2D1::RectF(0, 0, mSize.width, mSize.height); for (auto iter = mPushedClips.begin();iter != mPushedClips.end(); iter++) { if (iter->mPath) { return false; } aClipRect = IntersectRect(aClipRect, iter->mBounds); if (!iter->mIsPixelAligned) { aIsPixelAligned = false; } } return true; } already_AddRefed DrawTargetD2D1::GetClippedGeometry(IntRect *aClipBounds) { if (mCurrentClippedGeometry) { *aClipBounds = mCurrentClipBounds; RefPtr clippedGeometry(mCurrentClippedGeometry); return clippedGeometry.forget(); } MOZ_ASSERT(mPushedClips.size()); mCurrentClipBounds = IntRect(IntPoint(0, 0), mSize); // if pathGeom is null then pathRect represents the path. RefPtr pathGeom; D2D1_RECT_F pathRect; bool pathRectIsAxisAligned = false; auto iter = mPushedClips.begin(); if (iter->mPath) { pathGeom = GetTransformedGeometry(iter->mPath->GetGeometry(), iter->mTransform); } else { pathRect = iter->mBounds; pathRectIsAxisAligned = iter->mIsPixelAligned; } iter++; for (;iter != mPushedClips.end(); iter++) { // Do nothing but add it to the current clip bounds. if (!iter->mPath && iter->mIsPixelAligned) { mCurrentClipBounds.IntersectRect(mCurrentClipBounds, IntRect(int32_t(iter->mBounds.left), int32_t(iter->mBounds.top), int32_t(iter->mBounds.right - iter->mBounds.left), int32_t(iter->mBounds.bottom - iter->mBounds.top))); continue; } if (!pathGeom) { if (pathRectIsAxisAligned) { mCurrentClipBounds.IntersectRect(mCurrentClipBounds, IntRect(int32_t(pathRect.left), int32_t(pathRect.top), int32_t(pathRect.right - pathRect.left), int32_t(pathRect.bottom - pathRect.top))); } if (iter->mPath) { // See if pathRect needs to go into the path geometry. if (!pathRectIsAxisAligned) { pathGeom = ConvertRectToGeometry(pathRect); } else { pathGeom = GetTransformedGeometry(iter->mPath->GetGeometry(), iter->mTransform); } } else { pathRect = IntersectRect(pathRect, iter->mBounds); pathRectIsAxisAligned = false; continue; } } RefPtr newGeom; factory()->CreatePathGeometry(byRef(newGeom)); RefPtr currentSink; newGeom->Open(byRef(currentSink)); if (iter->mPath) { pathGeom->CombineWithGeometry(iter->mPath->GetGeometry(), D2D1_COMBINE_MODE_INTERSECT, iter->mTransform, currentSink); } else { RefPtr rectGeom = ConvertRectToGeometry(iter->mBounds); pathGeom->CombineWithGeometry(rectGeom, D2D1_COMBINE_MODE_INTERSECT, D2D1::IdentityMatrix(), currentSink); } currentSink->Close(); pathGeom = newGeom.forget(); } // For now we need mCurrentClippedGeometry to always be non-nullptr. This // method might seem a little strange but it is just fine, if pathGeom is // nullptr pathRect will always still contain 1 clip unaccounted for // regardless of mCurrentClipBounds. if (!pathGeom) { pathGeom = ConvertRectToGeometry(pathRect); } mCurrentClippedGeometry = pathGeom.forget(); *aClipBounds = mCurrentClipBounds; RefPtr clippedGeometry(mCurrentClippedGeometry); return clippedGeometry.forget(); } already_AddRefed DrawTargetD2D1::GetInverseClippedGeometry() { IntRect bounds; RefPtr geom = GetClippedGeometry(&bounds); RefPtr rectGeom; RefPtr inverseGeom; factory()->CreateRectangleGeometry(D2D1::RectF(0, 0, mSize.width, mSize.height), byRef(rectGeom)); factory()->CreatePathGeometry(byRef(inverseGeom)); RefPtr sink; inverseGeom->Open(byRef(sink)); rectGeom->CombineWithGeometry(geom, D2D1_COMBINE_MODE_EXCLUDE, D2D1::IdentityMatrix(), sink); sink->Close(); return inverseGeom.forget(); } void DrawTargetD2D1::PopAllClips() { if (mClipsArePushed) { PopClipsFromDC(mDC); mClipsArePushed = false; } } void DrawTargetD2D1::PushAllClips() { if (!mClipsArePushed) { PushClipsToDC(mDC); mClipsArePushed = true; } } void DrawTargetD2D1::PushClipsToDC(ID2D1DeviceContext *aDC, bool aForceIgnoreAlpha, const D2D1_RECT_F& aMaxRect) { mDC->SetTransform(D2D1::IdentityMatrix()); mTransformDirty = true; for (std::vector::iterator iter = mPushedClips.begin(); iter != mPushedClips.end(); iter++) { if (iter->mPath) { PushD2DLayer(aDC, iter->mPath->mGeometry, iter->mTransform, aForceIgnoreAlpha, aMaxRect); } else { mDC->PushAxisAlignedClip(iter->mBounds, iter->mIsPixelAligned ? D2D1_ANTIALIAS_MODE_ALIASED : D2D1_ANTIALIAS_MODE_PER_PRIMITIVE); } } } void DrawTargetD2D1::PopClipsFromDC(ID2D1DeviceContext *aDC) { for (int i = mPushedClips.size() - 1; i >= 0; i--) { if (mPushedClips[i].mPath) { aDC->PopLayer(); } else { aDC->PopAxisAlignedClip(); } } } already_AddRefed DrawTargetD2D1::CreateTransparentBlackBrush() { return GetSolidColorBrush(D2D1::ColorF(0, 0)); } already_AddRefed DrawTargetD2D1::GetSolidColorBrush(const D2D_COLOR_F& aColor) { RefPtr brush = mSolidColorBrush; brush->SetColor(aColor); return brush.forget(); } already_AddRefed DrawTargetD2D1::CreateBrushForPattern(const Pattern &aPattern, Float aAlpha) { if (!IsPatternSupportedByD2D(aPattern)) { return GetSolidColorBrush(D2D1::ColorF(1.0f, 1.0f, 1.0f, 1.0f)); } if (aPattern.GetType() == PatternType::COLOR) { Color color = static_cast(&aPattern)->mColor; return GetSolidColorBrush(D2D1::ColorF(color.r, color.g, color.b, color.a * aAlpha)); } if (aPattern.GetType() == PatternType::LINEAR_GRADIENT) { RefPtr gradBrush; const LinearGradientPattern *pat = static_cast(&aPattern); GradientStopsD2D *stops = static_cast(pat->mStops.get()); if (!stops) { gfxDebug() << "No stops specified for gradient pattern."; return CreateTransparentBlackBrush(); } if (pat->mBegin == pat->mEnd) { uint32_t stopCount = stops->mStopCollection->GetGradientStopCount(); vector d2dStops(stopCount); stops->mStopCollection->GetGradientStops(&d2dStops.front(), stopCount); d2dStops.back().color.a *= aAlpha; return GetSolidColorBrush(d2dStops.back().color); } mDC->CreateLinearGradientBrush(D2D1::LinearGradientBrushProperties(D2DPoint(pat->mBegin), D2DPoint(pat->mEnd)), D2D1::BrushProperties(aAlpha, D2DMatrix(pat->mMatrix)), stops->mStopCollection, byRef(gradBrush)); if (!gradBrush) { gfxWarning() << "Couldn't create gradient brush."; return CreateTransparentBlackBrush(); } return gradBrush.forget(); } if (aPattern.GetType() == PatternType::RADIAL_GRADIENT) { RefPtr gradBrush; const RadialGradientPattern *pat = static_cast(&aPattern); GradientStopsD2D *stops = static_cast(pat->mStops.get()); if (!stops) { gfxDebug() << "No stops specified for gradient pattern."; return CreateTransparentBlackBrush(); } // This will not be a complex radial gradient brush. mDC->CreateRadialGradientBrush( D2D1::RadialGradientBrushProperties(D2DPoint(pat->mCenter2), D2DPoint(pat->mCenter1 - pat->mCenter2), pat->mRadius2, pat->mRadius2), D2D1::BrushProperties(aAlpha, D2DMatrix(pat->mMatrix)), stops->mStopCollection, byRef(gradBrush)); if (!gradBrush) { gfxWarning() << "Couldn't create gradient brush."; return CreateTransparentBlackBrush(); } return gradBrush.forget(); } if (aPattern.GetType() == PatternType::SURFACE) { const SurfacePattern *pat = static_cast(&aPattern); if (!pat->mSurface) { gfxDebug() << "No source surface specified for surface pattern"; return CreateTransparentBlackBrush(); } D2D1_RECT_F samplingBounds; Matrix mat = pat->mMatrix; MOZ_ASSERT(pat->mSurface->IsValid()); RefPtr image = GetImageForSurface(pat->mSurface, mat, pat->mExtendMode, !pat->mSamplingRect.IsEmpty() ? &pat->mSamplingRect : nullptr); if (!image) { return CreateTransparentBlackBrush(); } bool useSamplingRect = false; if (pat->mSamplingRect.IsEmpty()) { RefPtr bitmap; image->QueryInterface((ID2D1Bitmap**)byRef(bitmap)); if (bitmap) { RefPtr bitmapBrush; mDC->CreateBitmapBrush(bitmap, D2D1::BitmapBrushProperties(D2DExtend(pat->mExtendMode), D2DExtend(pat->mExtendMode), D2DFilter(pat->mFilter)), D2D1::BrushProperties(aAlpha, D2DMatrix(mat)), byRef(bitmapBrush)); if (!bitmapBrush) { gfxWarning() << "Couldn't create bitmap brush!"; return CreateTransparentBlackBrush(); } return bitmapBrush.forget(); } } RefPtr imageBrush; if (pat->mSamplingRect.IsEmpty()) { samplingBounds = D2D1::RectF(0, 0, Float(pat->mSurface->GetSize().width), Float(pat->mSurface->GetSize().height)); } else if (pat->mSurface->GetType() == SurfaceType::D2D1_1_IMAGE) { samplingBounds = D2DRect(pat->mSamplingRect); mat.PreTranslate(pat->mSamplingRect.x, pat->mSamplingRect.y); } else { // We will do a partial upload of the sampling restricted area from GetImageForSurface. samplingBounds = D2D1::RectF(0, 0, pat->mSamplingRect.width, pat->mSamplingRect.height); } mDC->CreateImageBrush(image, D2D1::ImageBrushProperties(samplingBounds, D2DExtend(pat->mExtendMode), D2DExtend(pat->mExtendMode), D2DInterpolationMode(pat->mFilter)), D2D1::BrushProperties(aAlpha, D2DMatrix(mat)), byRef(imageBrush)); if (!imageBrush) { gfxWarning() << "Couldn't create image brush!"; return CreateTransparentBlackBrush(); } return imageBrush.forget(); } gfxWarning() << "Invalid pattern type detected."; return CreateTransparentBlackBrush(); } already_AddRefed DrawTargetD2D1::GetImageForSurface(SourceSurface *aSurface, Matrix &aSourceTransform, ExtendMode aExtendMode, const IntRect* aSourceRect) { RefPtr image; switch (aSurface->GetType()) { case SurfaceType::D2D1_1_IMAGE: { SourceSurfaceD2D1 *surf = static_cast(aSurface); image = surf->GetImage(); AddDependencyOnSource(surf); } break; default: { RefPtr dataSurf = aSurface->GetDataSurface(); if (!dataSurf) { gfxWarning() << "Invalid surface type."; return nullptr; } return CreatePartialBitmapForSurface(dataSurf, mTransform, mSize, aExtendMode, aSourceTransform, mDC, aSourceRect); } break; } return image.forget(); } already_AddRefed DrawTargetD2D1::OptimizeSourceSurface(SourceSurface* aSurface) const { if (aSurface->GetType() == SurfaceType::D2D1_1_IMAGE) { RefPtr surface(aSurface); return surface.forget(); } RefPtr data = aSurface->GetDataSurface(); RefPtr bitmap; { DataSourceSurface::ScopedMap map(data, DataSourceSurface::READ); if (MOZ2D_WARN_IF(!map.IsMapped())) { return nullptr; } HRESULT hr = mDC->CreateBitmap(D2DIntSize(data->GetSize()), map.GetData(), map.GetStride(), D2D1::BitmapProperties1(D2D1_BITMAP_OPTIONS_NONE, D2DPixelFormat(data->GetFormat())), byRef(bitmap)); if (FAILED(hr)) { gfxCriticalError(CriticalLog::DefaultOptions(Factory::ReasonableSurfaceSize(data->GetSize()))) << "[D2D1.1] 4CreateBitmap failure " << data->GetSize() << " Code: " << hexa(hr) << " format " << (int)data->GetFormat(); } } if (!bitmap) { return data.forget(); } return MakeAndAddRef(bitmap.get(), mDC, data->GetFormat(), data->GetSize()); } void DrawTargetD2D1::PushD2DLayer(ID2D1DeviceContext *aDC, ID2D1Geometry *aGeometry, const D2D1_MATRIX_3X2_F &aTransform, bool aForceIgnoreAlpha, const D2D1_RECT_F& aMaxRect) { D2D1_LAYER_OPTIONS1 options = D2D1_LAYER_OPTIONS1_NONE; if (aDC->GetPixelFormat().alphaMode == D2D1_ALPHA_MODE_IGNORE || aForceIgnoreAlpha) { options = D2D1_LAYER_OPTIONS1_IGNORE_ALPHA | D2D1_LAYER_OPTIONS1_INITIALIZE_FROM_BACKGROUND; } mDC->PushLayer(D2D1::LayerParameters1(aMaxRect, aGeometry, D2D1_ANTIALIAS_MODE_PER_PRIMITIVE, aTransform, 1.0, nullptr, options), nullptr); } } }