1 /*
2  * Copyright 2014 Google Inc.
3  *
4  * Use of this source code is governed by a BSD-style license that can be
5  * found in the LICENSE file.
6  */
7 
8 #include "SkLocalMatrixShader.h"
9 #include "SkTLazy.h"
10 
11 #if SK_SUPPORT_GPU
12 #include "GrFragmentProcessor.h"
13 #endif
14 
15 #if SK_SUPPORT_GPU
16 std::unique_ptr<GrFragmentProcessor> SkLocalMatrixShader::asFragmentProcessor(
17         const GrFPArgs& args) const {
18     return as_SB(fProxyShader)->asFragmentProcessor(
19         GrFPArgs::WithPreLocalMatrix(args, this->getLocalMatrix()));
20 }
21 #endif
22 
23 sk_sp<SkFlattenable> SkLocalMatrixShader::CreateProc(SkReadBuffer& buffer) {
24     SkMatrix lm;
25     buffer.readMatrix(&lm);
26     auto baseShader(buffer.readShader());
27     if (!baseShader) {
28         return nullptr;
29     }
30     return baseShader->makeWithLocalMatrix(lm);
31 }
32 
33 void SkLocalMatrixShader::flatten(SkWriteBuffer& buffer) const {
34     buffer.writeMatrix(this->getLocalMatrix());
35     buffer.writeFlattenable(fProxyShader.get());
36 }
37 
38 #ifdef SK_ENABLE_LEGACY_SHADERCONTEXT
39 SkShaderBase::Context* SkLocalMatrixShader::onMakeContext(
40     const ContextRec& rec, SkArenaAlloc* alloc) const
41 {
42     SkTCopyOnFirstWrite<SkMatrix> lm(this->getLocalMatrix());
43     if (rec.fLocalMatrix) {
44         lm.writable()->preConcat(*rec.fLocalMatrix);
45     }
46 
47     ContextRec newRec(rec);
48     newRec.fLocalMatrix = lm;
49 
50     return as_SB(fProxyShader)->makeContext(newRec, alloc);
51 }
52 #endif
53 
54 SkImage* SkLocalMatrixShader::onIsAImage(SkMatrix* outMatrix, enum TileMode* mode) const {
55     SkMatrix imageMatrix;
56     SkImage* image = fProxyShader->isAImage(&imageMatrix, mode);
57     if (image && outMatrix) {
58         // Local matrix must be applied first so it is on the right side of the concat.
59         *outMatrix = SkMatrix::Concat(imageMatrix, this->getLocalMatrix());
60     }
61 
62     return image;
63 }
64 
65 bool SkLocalMatrixShader::onAppendStages(const StageRec& rec) const {
66     SkTCopyOnFirstWrite<SkMatrix> lm(this->getLocalMatrix());
67     if (rec.fLocalM) {
68         lm.writable()->preConcat(*rec.fLocalM);
69     }
70 
71     StageRec newRec = rec;
72     newRec.fLocalM = lm;
73     return as_SB(fProxyShader)->appendStages(newRec);
74 }
75 
76 sk_sp<SkShader> SkShader::makeWithLocalMatrix(const SkMatrix& localMatrix) const {
77     if (localMatrix.isIdentity()) {
78         return sk_ref_sp(const_cast<SkShader*>(this));
79     }
80 
81     const SkMatrix* lm = &localMatrix;
82 
83     sk_sp<SkShader> baseShader;
84     SkMatrix otherLocalMatrix;
85     sk_sp<SkShader> proxy(as_SB(this)->makeAsALocalMatrixShader(&otherLocalMatrix));
86     if (proxy) {
87         otherLocalMatrix.preConcat(localMatrix);
88         lm = &otherLocalMatrix;
89         baseShader = proxy;
90     } else {
91         baseShader = sk_ref_sp(const_cast<SkShader*>(this));
92     }
93 
94     return sk_make_sp<SkLocalMatrixShader>(std::move(baseShader), *lm);
95 }
96