/*
* Copyright (C) 2016 The Android Open Source Project
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
#include <VectorDrawable.h>
#include <gtest/gtest.h>
#include <SkClipStack.h>
#include <SkLiteRecorder.h>
#include <SkSurface_Base.h>
#include <string.h>
#include "AnimationContext.h"
#include "DamageAccumulator.h"
#include "FatalTestCanvas.h"
#include "IContextFactory.h"
#include "SkiaCanvas.h"
#include "pipeline/skia/SkiaDisplayList.h"
#include "pipeline/skia/SkiaPipeline.h"
#include "pipeline/skia/SkiaRecordingCanvas.h"
#include "renderthread/CanvasContext.h"
#include "tests/common/TestUtils.h"
using namespace android;
using namespace android::uirenderer;
using namespace android::uirenderer::renderthread;
using namespace android::uirenderer::skiapipeline;
namespace {
static void testProperty(std::function<void(RenderProperties&)> propSetupCallback,
std::function<void(const SkCanvas&)> opValidateCallback) {
static const int CANVAS_WIDTH = 100;
static const int CANVAS_HEIGHT = 100;
class PropertyTestCanvas : public TestCanvasBase {
public:
PropertyTestCanvas(std::function<void(const SkCanvas&)> callback)
: TestCanvasBase(CANVAS_WIDTH, CANVAS_HEIGHT), mCallback(callback) {}
void onDrawRect(const SkRect& rect, const SkPaint& paint) override {
EXPECT_EQ(mDrawCounter++, 0);
mCallback(*this);
}
void onClipRRect(const SkRRect& rrect, SkClipOp op, ClipEdgeStyle style) {
SkCanvas::onClipRRect(rrect, op, style);
}
std::function<void(const SkCanvas&)> mCallback;
};
auto node = TestUtils::createSkiaNode(
0, 0, CANVAS_WIDTH, CANVAS_HEIGHT,
[propSetupCallback](RenderProperties& props, SkiaRecordingCanvas& canvas) {
propSetupCallback(props);
SkPaint paint;
paint.setColor(SK_ColorWHITE);
canvas.drawRect(0, 0, CANVAS_WIDTH, CANVAS_HEIGHT, paint);
});
PropertyTestCanvas canvas(opValidateCallback);
RenderNodeDrawable drawable(node.get(), &canvas, true);
canvas.drawDrawable(&drawable);
EXPECT_EQ(1, canvas.mDrawCounter);
}
}
TEST(RenderNodeDrawable, renderPropClipping) {
testProperty(
[](RenderProperties& properties) {
properties.setClipToBounds(true);
properties.setClipBounds(android::uirenderer::Rect(10, 20, 300, 400));
},
[](const SkCanvas& canvas) {
EXPECT_EQ(SkRect::MakeLTRB(10, 20, 100, 100), TestUtils::getClipBounds(&canvas))
<< "Clip rect should be intersection of node bounds and clip bounds";
});
}
TEST(RenderNodeDrawable, renderPropRevealClip) {
testProperty(
[](RenderProperties& properties) {
properties.mutableRevealClip().set(true, 50, 50, 25);
},
[](const SkCanvas& canvas) {
EXPECT_EQ(SkRect::MakeLTRB(25, 25, 75, 75), TestUtils::getClipBounds(&canvas));
});
}
TEST(RenderNodeDrawable, renderPropOutlineClip) {
testProperty(
[](RenderProperties& properties) {
properties.mutableOutline().setShouldClip(true);
properties.mutableOutline().setRoundRect(10, 20, 30, 40, 5.0f, 0.5f);
},
[](const SkCanvas& canvas) {
EXPECT_EQ(SkRect::MakeLTRB(10, 20, 30, 40), TestUtils::getClipBounds(&canvas));
});
}
TEST(RenderNodeDrawable, renderPropTransform) {
testProperty(
[](RenderProperties& properties) {
properties.setLeftTopRightBottom(10, 10, 110, 110);
SkMatrix staticMatrix = SkMatrix::MakeScale(1.2f, 1.2f);
properties.setStaticMatrix(&staticMatrix);
// ignored, since static overrides animation
SkMatrix animationMatrix = SkMatrix::MakeTrans(15, 15);
properties.setAnimationMatrix(&animationMatrix);
properties.setTranslationX(10);
properties.setTranslationY(20);
properties.setScaleX(0.5f);
properties.setScaleY(0.7f);
},
[](const SkCanvas& canvas) {
Matrix4 matrix;
matrix.loadTranslate(10, 10, 0); // left, top
matrix.scale(1.2f, 1.2f, 1); // static matrix
// ignore animation matrix, since static overrides it
// translation xy
matrix.translate(10, 20);
// scale xy (from default pivot - center)
matrix.translate(50, 50);
matrix.scale(0.5f, 0.7f, 1);
matrix.translate(-50, -50);
Matrix4 actual(canvas.getTotalMatrix());
EXPECT_MATRIX_APPROX_EQ(matrix, actual) << "Op draw matrix must match expected "
"combination of transformation "
"properties";
});
}