/* * Copyright 2017 Google Inc. * * Use of this source code is governed by a BSD-style license that can be * found in the LICENSE file. */ #include "SkSGTransform.h" #include "SkCanvas.h" namespace sksg { // Matrix nodes don't generate damage on their own, but via aggregation ancestor Transform nodes. Matrix::Matrix(const SkMatrix& m, sk_sp<Matrix> parent) : INHERITED(kBubbleDamage_Trait) , fParent(std::move(parent)) , fLocalMatrix(m) { if (fParent) { this->observeInval(fParent); } } Matrix::~Matrix() { if (fParent) { this->unobserveInval(fParent); } } SkRect Matrix::onRevalidate(InvalidationController* ic, const SkMatrix& ctm) { fTotalMatrix = fLocalMatrix; if (fParent) { fParent->revalidate(ic, ctm); fTotalMatrix.postConcat(fParent->getTotalMatrix()); } return SkRect::MakeEmpty(); } Transform::Transform(sk_sp<RenderNode> child, sk_sp<Matrix> matrix) : INHERITED(std::move(child)) , fMatrix(std::move(matrix)) { this->observeInval(fMatrix); } Transform::~Transform() { this->unobserveInval(fMatrix); } void Transform::onRender(SkCanvas* canvas) const { const auto& m = fMatrix->getTotalMatrix(); SkAutoCanvasRestore acr(canvas, !m.isIdentity()); canvas->concat(m); this->INHERITED::onRender(canvas); } SkRect Transform::onRevalidate(InvalidationController* ic, const SkMatrix& ctm) { SkASSERT(this->hasInval()); // We don't care about matrix reval results. fMatrix->revalidate(ic, ctm); const auto& m = fMatrix->getTotalMatrix(); auto bounds = this->INHERITED::onRevalidate(ic, SkMatrix::Concat(ctm, m)); m.mapRect(&bounds); return bounds; } } // namespace sksg