2016-07-27 01:46:34 +00:00
|
|
|
/*
|
|
|
|
* Copyright 2016 Google Inc.
|
|
|
|
*
|
|
|
|
* Use of this source code is governed by a BSD-style license that can be
|
|
|
|
* found in the LICENSE file.
|
|
|
|
*/
|
|
|
|
|
2019-04-23 17:05:21 +00:00
|
|
|
#include "experimental/svg/model/SkSVGRenderContext.h"
|
|
|
|
#include "experimental/svg/model/SkSVGSVG.h"
|
|
|
|
#include "experimental/svg/model/SkSVGValue.h"
|
|
|
|
#include "include/core/SkCanvas.h"
|
2016-07-27 01:46:34 +00:00
|
|
|
|
2016-07-29 15:52:03 +00:00
|
|
|
SkSVGSVG::SkSVGSVG() : INHERITED(SkSVGTag::kSvg) { }
|
2016-08-03 17:21:11 +00:00
|
|
|
|
2016-08-08 18:38:55 +00:00
|
|
|
bool SkSVGSVG::onPrepareToRender(SkSVGRenderContext* ctx) const {
|
|
|
|
auto viewPortRect = ctx->lengthContext().resolveRect(fX, fY, fWidth, fHeight);
|
|
|
|
auto contentMatrix = SkMatrix::MakeTrans(viewPortRect.x(), viewPortRect.y());
|
|
|
|
auto viewPort = SkSize::Make(viewPortRect.width(), viewPortRect.height());
|
|
|
|
|
|
|
|
if (fViewBox.isValid()) {
|
|
|
|
const SkRect& viewBox = *fViewBox.get();
|
|
|
|
|
|
|
|
// An empty viewbox disables rendering.
|
|
|
|
if (viewBox.isEmpty()) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// A viewBox overrides the intrinsic viewport.
|
|
|
|
viewPort = SkSize::Make(viewBox.width(), viewBox.height());
|
|
|
|
|
|
|
|
contentMatrix.preConcat(
|
|
|
|
SkMatrix::MakeRectToRect(viewBox, viewPortRect, SkMatrix::kFill_ScaleToFit));
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!contentMatrix.isIdentity()) {
|
2017-10-11 18:11:16 +00:00
|
|
|
ctx->saveOnce();
|
2016-08-08 18:38:55 +00:00
|
|
|
ctx->canvas()->concat(contentMatrix);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (viewPort != ctx->lengthContext().viewPort()) {
|
|
|
|
ctx->writableLengthContext()->setViewPort(viewPort);
|
|
|
|
}
|
|
|
|
|
|
|
|
return this->INHERITED::onPrepareToRender(ctx);
|
|
|
|
}
|
|
|
|
|
2016-08-03 17:21:11 +00:00
|
|
|
void SkSVGSVG::setX(const SkSVGLength& x) {
|
|
|
|
fX = x;
|
|
|
|
}
|
|
|
|
|
|
|
|
void SkSVGSVG::setY(const SkSVGLength& y) {
|
|
|
|
fY = y;
|
|
|
|
}
|
|
|
|
|
|
|
|
void SkSVGSVG::setWidth(const SkSVGLength& w) {
|
|
|
|
fWidth = w;
|
|
|
|
}
|
|
|
|
|
|
|
|
void SkSVGSVG::setHeight(const SkSVGLength& h) {
|
|
|
|
fHeight = h;
|
|
|
|
}
|
|
|
|
|
2016-08-08 18:38:55 +00:00
|
|
|
void SkSVGSVG::setViewBox(const SkSVGViewBoxType& vb) {
|
|
|
|
fViewBox.set(vb);
|
|
|
|
}
|
|
|
|
|
2016-08-03 17:21:11 +00:00
|
|
|
void SkSVGSVG::onSetAttribute(SkSVGAttribute attr, const SkSVGValue& v) {
|
|
|
|
switch (attr) {
|
|
|
|
case SkSVGAttribute::kX:
|
|
|
|
if (const auto* x = v.as<SkSVGLengthValue>()) {
|
|
|
|
this->setX(*x);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case SkSVGAttribute::kY:
|
|
|
|
if (const auto* y = v.as<SkSVGLengthValue>()) {
|
|
|
|
this->setY(*y);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case SkSVGAttribute::kWidth:
|
|
|
|
if (const auto* w = v.as<SkSVGLengthValue>()) {
|
|
|
|
this->setWidth(*w);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case SkSVGAttribute::kHeight:
|
|
|
|
if (const auto* h = v.as<SkSVGLengthValue>()) {
|
|
|
|
this->setHeight(*h);
|
|
|
|
}
|
|
|
|
break;
|
2016-08-08 18:38:55 +00:00
|
|
|
case SkSVGAttribute::kViewBox:
|
|
|
|
if (const auto* vb = v.as<SkSVGViewBoxValue>()) {
|
|
|
|
this->setViewBox(*vb);
|
|
|
|
}
|
|
|
|
break;
|
2016-08-03 17:21:11 +00:00
|
|
|
default:
|
|
|
|
this->INHERITED::onSetAttribute(attr, v);
|
|
|
|
}
|
|
|
|
}
|
2016-09-14 19:04:30 +00:00
|
|
|
|
2020-04-08 14:10:53 +00:00
|
|
|
// https://www.w3.org/TR/SVG11/coords.html#IntrinsicSizing
|
2016-09-14 19:04:30 +00:00
|
|
|
SkSize SkSVGSVG::intrinsicSize(const SkSVGLengthContext& lctx) const {
|
|
|
|
// Percentage values do not provide an intrinsic size.
|
|
|
|
if (fWidth.unit() == SkSVGLength::Unit::kPercentage ||
|
|
|
|
fHeight.unit() == SkSVGLength::Unit::kPercentage) {
|
|
|
|
return SkSize::Make(0, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
return SkSize::Make(lctx.resolve(fWidth, SkSVGLengthContext::LengthType::kHorizontal),
|
|
|
|
lctx.resolve(fHeight, SkSVGLengthContext::LengthType::kVertical));
|
|
|
|
}
|