blob: 896c01ce910c57a1fde2f5f6d5aa2d05d95fd18e [file] [log] [blame]
Stan Iliev021693b2016-10-17 16:26:15 -04001/*
2 * Copyright (C) 2016 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17#include "LayerDrawable.h"
John Reck1bcacfd2017-11-03 10:12:19 -070018#include "GlLayer.h"
Greg Daniel45ec62b2017-01-04 14:27:00 -050019#include "VkLayer.h"
20
Greg Danielac2d2322017-07-12 11:30:15 -040021#include "GrBackendSurface.h"
Derek Sollenbergerf87da672016-11-02 11:34:27 -040022#include "SkColorFilter.h"
Greg Daniel45ec62b2017-01-04 14:27:00 -050023#include "SkSurface.h"
Stan Iliev021693b2016-10-17 16:26:15 -040024#include "gl/GrGLTypes.h"
25
26namespace android {
27namespace uirenderer {
28namespace skiapipeline {
29
30void LayerDrawable::onDraw(SkCanvas* canvas) {
Derek Sollenbergerf5a370e2017-06-15 13:50:08 -040031 Layer* layer = mLayerUpdater->backingLayer();
32 if (layer) {
33 DrawLayer(canvas->getGrContext(), canvas, layer);
34 }
Derek Sollenbergerc4fbada2016-11-07 16:05:41 -050035}
36
Leon Scroggins III1a12ab22018-03-26 15:00:49 -040037bool LayerDrawable::DrawLayer(GrContext* context, SkCanvas* canvas, Layer* layer,
38 const SkRect* dstRect) {
Stan Ilieve9d00122017-09-19 12:07:10 -040039 if (context == nullptr) {
40 SkDEBUGF(("Attempting to draw LayerDrawable into an unsupported surface"));
41 return false;
42 }
Stan Iliev021693b2016-10-17 16:26:15 -040043 // transform the matrix based on the layer
Stan Iliev944dbf22017-09-27 11:05:29 -040044 SkMatrix layerTransform;
45 layer->getTransform().copyTo(layerTransform);
Greg Daniel45ec62b2017-01-04 14:27:00 -050046 sk_sp<SkImage> layerImage;
Leon Scroggins III1a12ab22018-03-26 15:00:49 -040047 const int layerWidth = layer->getWidth();
48 const int layerHeight = layer->getHeight();
Greg Daniel45ec62b2017-01-04 14:27:00 -050049 if (layer->getApi() == Layer::Api::OpenGL) {
50 GlLayer* glLayer = static_cast<GlLayer*>(layer);
51 GrGLTextureInfo externalTexture;
52 externalTexture.fTarget = glLayer->getRenderTarget();
Greg Daniel03d50fa2018-03-20 11:00:20 -040053 SkASSERT(GL_RGBA == glLayer->getTexture().internalFormat());
54 externalTexture.fFormat = GL_RGBA8;
Greg Daniel45ec62b2017-01-04 14:27:00 -050055 externalTexture.fID = glLayer->getTextureId();
Greg Daniel03d50fa2018-03-20 11:00:20 -040056 GrBackendTexture backendTexture(layerWidth, layerHeight, GrMipMapped::kNo, externalTexture);
Greg Danielac2d2322017-07-12 11:30:15 -040057 layerImage = SkImage::MakeFromTexture(context, backendTexture, kTopLeft_GrSurfaceOrigin,
Greg Daniel03d50fa2018-03-20 11:00:20 -040058 kRGBA_8888_SkColorType, kPremul_SkAlphaType, nullptr);
Greg Daniel45ec62b2017-01-04 14:27:00 -050059 } else {
60 SkASSERT(layer->getApi() == Layer::Api::Vulkan);
61 VkLayer* vkLayer = static_cast<VkLayer*>(layer);
62 canvas->clear(SK_ColorGREEN);
63 layerImage = vkLayer->getImage();
64 }
65
Stan Iliev021693b2016-10-17 16:26:15 -040066 if (layerImage) {
Leon Scroggins III1a12ab22018-03-26 15:00:49 -040067 SkMatrix textureMatrixInv;
68 layer->getTexTransform().copyTo(textureMatrixInv);
John Reck1bcacfd2017-11-03 10:12:19 -070069 // TODO: after skia bug https://bugs.chromium.org/p/skia/issues/detail?id=7075 is fixed
Stan Iliev944dbf22017-09-27 11:05:29 -040070 // use bottom left origin and remove flipV and invert transformations.
71 SkMatrix flipV;
72 flipV.setAll(1, 0, 0, 0, -1, 1, 0, 0, 1);
Leon Scroggins III1a12ab22018-03-26 15:00:49 -040073 textureMatrixInv.preConcat(flipV);
74 textureMatrixInv.preScale(1.0f / layerWidth, 1.0f / layerHeight);
75 textureMatrixInv.postScale(layerWidth, layerHeight);
76 SkMatrix textureMatrix;
77 if (!textureMatrixInv.invert(&textureMatrix)) {
78 textureMatrix = textureMatrixInv;
Stan Iliev944dbf22017-09-27 11:05:29 -040079 }
80
Leon Scroggins III1a12ab22018-03-26 15:00:49 -040081 SkMatrix matrix = SkMatrix::Concat(layerTransform, textureMatrix);
Stan Iliev944dbf22017-09-27 11:05:29 -040082
Stan Iliev021693b2016-10-17 16:26:15 -040083 SkPaint paint;
Derek Sollenbergerc4fbada2016-11-07 16:05:41 -050084 paint.setAlpha(layer->getAlpha());
85 paint.setBlendMode(layer->getMode());
86 paint.setColorFilter(sk_ref_sp(layer->getColorFilter()));
Stan Ilieva73b0be2017-10-06 10:16:58 -040087
88 const bool nonIdentityMatrix = !matrix.isIdentity();
89 if (nonIdentityMatrix) {
90 canvas->save();
91 canvas->concat(matrix);
92 }
Leon Scroggins III1a12ab22018-03-26 15:00:49 -040093 if (dstRect) {
94 SkMatrix matrixInv;
95 if (!matrix.invert(&matrixInv)) {
96 matrixInv = matrix;
97 }
98 SkRect srcRect = SkRect::MakeIWH(layerWidth, layerHeight);
99 matrixInv.mapRect(&srcRect);
100 SkRect skiaDestRect = *dstRect;
101 matrixInv.mapRect(&skiaDestRect);
102 canvas->drawImageRect(layerImage.get(), srcRect, skiaDestRect, &paint,
103 SkCanvas::kFast_SrcRectConstraint);
104 } else {
105 canvas->drawImage(layerImage.get(), 0, 0, &paint);
106 }
Stan Ilieva73b0be2017-10-06 10:16:58 -0400107 // restore the original matrix
108 if (nonIdentityMatrix) {
109 canvas->restore();
110 }
Stan Iliev021693b2016-10-17 16:26:15 -0400111 }
Derek Sollenbergerc4fbada2016-11-07 16:05:41 -0500112
113 return layerImage;
Stan Iliev021693b2016-10-17 16:26:15 -0400114}
115
John Reck1bcacfd2017-11-03 10:12:19 -0700116}; // namespace skiapipeline
117}; // namespace uirenderer
118}; // namespace android