| |
| /* |
| * Copyright 2013 Google Inc. |
| * |
| * Use of this source code is governed by a BSD-style license that can be |
| * found in the LICENSE file. |
| */ |
| |
| #include "GrPaint.h" |
| |
| #include "GrBlend.h" |
| #include "effects/GrSimpleTextureEffect.h" |
| |
| void GrPaint::addColorTextureProcessor(GrTexture* texture, const SkMatrix& matrix) { |
| this->addColorProcessor(GrSimpleTextureEffect::Create(texture, matrix))->unref(); |
| } |
| |
| void GrPaint::addCoverageTextureProcessor(GrTexture* texture, const SkMatrix& matrix) { |
| this->addCoverageProcessor(GrSimpleTextureEffect::Create(texture, matrix))->unref(); |
| } |
| |
| void GrPaint::addColorTextureProcessor(GrTexture* texture, |
| const SkMatrix& matrix, |
| const GrTextureParams& params) { |
| this->addColorProcessor(GrSimpleTextureEffect::Create(texture, matrix, params))->unref(); |
| } |
| |
| void GrPaint::addCoverageTextureProcessor(GrTexture* texture, |
| const SkMatrix& matrix, |
| const GrTextureParams& params) { |
| this->addCoverageProcessor(GrSimpleTextureEffect::Create(texture, matrix, params))->unref(); |
| } |
| |
| bool GrPaint::isOpaque() const { |
| return this->getOpaqueAndKnownColor(NULL, NULL); |
| } |
| |
| bool GrPaint::isOpaqueAndConstantColor(GrColor* color) const { |
| GrColor tempColor; |
| uint32_t colorComps; |
| if (this->getOpaqueAndKnownColor(&tempColor, &colorComps)) { |
| if (kRGBA_GrColorComponentFlags == colorComps) { |
| *color = tempColor; |
| return true; |
| } |
| } |
| return false; |
| } |
| |
| bool GrPaint::getOpaqueAndKnownColor(GrColor* solidColor, |
| uint32_t* solidColorKnownComponents) const { |
| |
| // TODO: Share this implementation with GrDrawState |
| |
| GrProcessor::InvariantOutput inout; |
| inout.fColor = GrColorPackRGBA(fCoverage, fCoverage, fCoverage, fCoverage); |
| inout.fValidFlags = kRGBA_GrColorComponentFlags; |
| inout.fIsSingleComponent = false; |
| int count = fCoverageStages.count(); |
| for (int i = 0; i < count; ++i) { |
| fCoverageStages[i].getProcessor()->computeInvariantOutput(&inout); |
| } |
| if (!inout.isSolidWhite()) { |
| return false; |
| } |
| |
| inout.fColor = fColor; |
| inout.fValidFlags = kRGBA_GrColorComponentFlags; |
| inout.fIsSingleComponent = false; |
| count = fColorStages.count(); |
| for (int i = 0; i < count; ++i) { |
| fColorStages[i].getProcessor()->computeInvariantOutput(&inout); |
| } |
| |
| SkASSERT((NULL == solidColor) == (NULL == solidColorKnownComponents)); |
| |
| GrBlendCoeff srcCoeff = fSrcBlendCoeff; |
| GrBlendCoeff dstCoeff = fDstBlendCoeff; |
| GrSimplifyBlend(&srcCoeff, &dstCoeff, inout.fColor, inout.fValidFlags, |
| 0, 0, 0); |
| |
| bool opaque = kZero_GrBlendCoeff == dstCoeff && !GrBlendCoeffRefsDst(srcCoeff); |
| if (solidColor) { |
| if (opaque) { |
| switch (srcCoeff) { |
| case kZero_GrBlendCoeff: |
| *solidColor = 0; |
| *solidColorKnownComponents = kRGBA_GrColorComponentFlags; |
| break; |
| |
| case kOne_GrBlendCoeff: |
| *solidColor = inout.fColor; |
| *solidColorKnownComponents = inout.fValidFlags; |
| break; |
| |
| // The src coeff should never refer to the src and if it refers to dst then opaque |
| // should have been false. |
| case kSC_GrBlendCoeff: |
| case kISC_GrBlendCoeff: |
| case kDC_GrBlendCoeff: |
| case kIDC_GrBlendCoeff: |
| case kSA_GrBlendCoeff: |
| case kISA_GrBlendCoeff: |
| case kDA_GrBlendCoeff: |
| case kIDA_GrBlendCoeff: |
| default: |
| SkFAIL("srcCoeff should not refer to src or dst."); |
| break; |
| |
| // TODO: update this once GrPaint actually has a const color. |
| case kConstC_GrBlendCoeff: |
| case kIConstC_GrBlendCoeff: |
| case kConstA_GrBlendCoeff: |
| case kIConstA_GrBlendCoeff: |
| *solidColorKnownComponents = 0; |
| break; |
| } |
| } else { |
| solidColorKnownComponents = 0; |
| } |
| } |
| return opaque; |
| } |