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/SkSVGNode.h"
|
|
|
|
#include "experimental/svg/model/SkSVGRenderContext.h"
|
2020-04-10 14:14:04 +00:00
|
|
|
#include "experimental/svg/model/SkSVGValue.h"
|
2019-04-23 17:05:21 +00:00
|
|
|
#include "include/core/SkCanvas.h"
|
|
|
|
#include "include/core/SkMatrix.h"
|
|
|
|
#include "include/pathops/SkPathOps.h"
|
|
|
|
#include "src/core/SkTLazy.h"
|
2016-07-27 01:46:34 +00:00
|
|
|
|
|
|
|
SkSVGNode::SkSVGNode(SkSVGTag t) : fTag(t) { }
|
|
|
|
|
|
|
|
SkSVGNode::~SkSVGNode() { }
|
|
|
|
|
2016-08-08 18:38:55 +00:00
|
|
|
void SkSVGNode::render(const SkSVGRenderContext& ctx) const {
|
|
|
|
SkSVGRenderContext localContext(ctx);
|
|
|
|
|
|
|
|
if (this->onPrepareToRender(&localContext)) {
|
|
|
|
this->onRender(localContext);
|
2016-07-27 01:46:34 +00:00
|
|
|
}
|
2016-08-08 18:38:55 +00:00
|
|
|
}
|
2016-07-27 01:46:34 +00:00
|
|
|
|
2016-09-13 00:06:47 +00:00
|
|
|
bool SkSVGNode::asPaint(const SkSVGRenderContext& ctx, SkPaint* paint) const {
|
|
|
|
SkSVGRenderContext localContext(ctx);
|
|
|
|
|
|
|
|
return this->onPrepareToRender(&localContext) && this->onAsPaint(localContext, paint);
|
|
|
|
}
|
|
|
|
|
2016-12-08 14:26:47 +00:00
|
|
|
SkPath SkSVGNode::asPath(const SkSVGRenderContext& ctx) const {
|
|
|
|
SkSVGRenderContext localContext(ctx);
|
2016-12-08 21:04:24 +00:00
|
|
|
if (!this->onPrepareToRender(&localContext)) {
|
|
|
|
return SkPath();
|
|
|
|
}
|
|
|
|
|
|
|
|
SkPath path = this->onAsPath(localContext);
|
|
|
|
|
|
|
|
if (const auto* clipPath = localContext.clipPath()) {
|
|
|
|
// There is a clip-path present on the current node.
|
|
|
|
Op(path, *clipPath, kIntersect_SkPathOp, &path);
|
|
|
|
}
|
|
|
|
|
|
|
|
return path;
|
2016-12-08 14:26:47 +00:00
|
|
|
}
|
|
|
|
|
2016-08-08 18:38:55 +00:00
|
|
|
bool SkSVGNode::onPrepareToRender(SkSVGRenderContext* ctx) const {
|
2016-09-20 22:45:57 +00:00
|
|
|
ctx->applyPresentationAttributes(fPresentationAttributes,
|
|
|
|
this->hasChildren() ? 0 : SkSVGRenderContext::kLeaf);
|
2017-10-12 15:33:28 +00:00
|
|
|
|
|
|
|
// visibility:hidden disables rendering
|
|
|
|
const auto visibility = ctx->presentationContext().fInherited.fVisibility.get()->type();
|
|
|
|
return visibility != SkSVGVisibility::Type::kHidden;
|
2016-07-27 01:46:34 +00:00
|
|
|
}
|
|
|
|
|
2020-04-10 14:14:04 +00:00
|
|
|
void SkSVGNode::setAttribute(SkSVGAttribute attr, const SkSVGValue& v) {
|
2016-07-27 01:46:34 +00:00
|
|
|
this->onSetAttribute(attr, v);
|
|
|
|
}
|
|
|
|
|
2016-12-08 14:26:47 +00:00
|
|
|
void SkSVGNode::setClipPath(const SkSVGClip& clip) {
|
|
|
|
fPresentationAttributes.fClipPath.set(clip);
|
|
|
|
}
|
|
|
|
|
2020-04-09 12:54:29 +00:00
|
|
|
template <typename T>
|
|
|
|
void SetInheritedByDefault(SkTLazy<T>& presentation_attribute, const T& value) {
|
|
|
|
if (value.type() != T::Type::kInherit) {
|
|
|
|
presentation_attribute.set(value);
|
|
|
|
} else {
|
|
|
|
// kInherited values are semantically equivalent to
|
|
|
|
// the absence of a local presentation attribute.
|
|
|
|
presentation_attribute.reset();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-10-10 15:22:08 +00:00
|
|
|
void SkSVGNode::setClipRule(const SkSVGFillRule& clipRule) {
|
2020-04-09 12:54:29 +00:00
|
|
|
SetInheritedByDefault(fPresentationAttributes.fClipRule, clipRule);
|
2017-10-10 15:22:08 +00:00
|
|
|
}
|
|
|
|
|
2020-04-09 18:14:10 +00:00
|
|
|
void SkSVGNode::setColor(const SkSVGColorType& color) {
|
|
|
|
// TODO: Color should be inherited by default
|
|
|
|
fPresentationAttributes.fColor.set(color);
|
|
|
|
}
|
|
|
|
|
2016-08-12 20:17:11 +00:00
|
|
|
void SkSVGNode::setFill(const SkSVGPaint& svgPaint) {
|
2020-04-09 12:54:29 +00:00
|
|
|
SetInheritedByDefault(fPresentationAttributes.fFill, svgPaint);
|
2016-08-12 20:17:11 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void SkSVGNode::setFillOpacity(const SkSVGNumberType& opacity) {
|
2020-04-09 18:36:45 +00:00
|
|
|
fPresentationAttributes.fFillOpacity.set(SkSVGNumberType(SkTPin<SkScalar>(opacity, 0, 1)));
|
2016-08-12 20:17:11 +00:00
|
|
|
}
|
|
|
|
|
2016-12-01 18:35:11 +00:00
|
|
|
void SkSVGNode::setFillRule(const SkSVGFillRule& fillRule) {
|
2020-04-09 12:54:29 +00:00
|
|
|
SetInheritedByDefault(fPresentationAttributes.fFillRule, fillRule);
|
2016-12-01 18:35:11 +00:00
|
|
|
}
|
|
|
|
|
2016-08-15 19:45:11 +00:00
|
|
|
void SkSVGNode::setOpacity(const SkSVGNumberType& opacity) {
|
2020-04-09 18:36:45 +00:00
|
|
|
fPresentationAttributes.fOpacity.set(SkSVGNumberType(SkTPin<SkScalar>(opacity, 0, 1)));
|
2016-08-15 19:45:11 +00:00
|
|
|
}
|
|
|
|
|
2016-08-12 20:17:11 +00:00
|
|
|
void SkSVGNode::setStroke(const SkSVGPaint& svgPaint) {
|
2020-04-09 12:54:29 +00:00
|
|
|
SetInheritedByDefault(fPresentationAttributes.fStroke, svgPaint);
|
2016-08-12 20:17:11 +00:00
|
|
|
}
|
|
|
|
|
2017-10-13 18:07:44 +00:00
|
|
|
void SkSVGNode::setStrokeDashArray(const SkSVGDashArray& dashArray) {
|
2020-04-09 12:54:29 +00:00
|
|
|
SetInheritedByDefault(fPresentationAttributes.fStrokeDashArray, dashArray);
|
2017-10-13 18:07:44 +00:00
|
|
|
}
|
|
|
|
|
2017-10-13 22:18:32 +00:00
|
|
|
void SkSVGNode::setStrokeDashOffset(const SkSVGLength& dashOffset) {
|
|
|
|
fPresentationAttributes.fStrokeDashOffset.set(dashOffset);
|
|
|
|
}
|
|
|
|
|
2016-08-12 20:17:11 +00:00
|
|
|
void SkSVGNode::setStrokeOpacity(const SkSVGNumberType& opacity) {
|
2020-04-09 18:36:45 +00:00
|
|
|
fPresentationAttributes.fStrokeOpacity.set(SkSVGNumberType(SkTPin<SkScalar>(opacity, 0, 1)));
|
2016-08-12 20:17:11 +00:00
|
|
|
}
|
|
|
|
|
2020-04-09 12:54:29 +00:00
|
|
|
void SkSVGNode::setStrokeLineCap(const SkSVGLineCap& lc) {
|
|
|
|
SetInheritedByDefault(fPresentationAttributes.fStrokeLineCap, lc);
|
|
|
|
}
|
|
|
|
|
|
|
|
void SkSVGNode::setStrokeLineJoin(const SkSVGLineJoin& lj) {
|
|
|
|
SetInheritedByDefault(fPresentationAttributes.fStrokeLineJoin, lj);
|
|
|
|
}
|
|
|
|
|
|
|
|
void SkSVGNode::setStrokeMiterLimit(const SkSVGNumberType& ml) {
|
|
|
|
fPresentationAttributes.fStrokeMiterLimit.set(ml);
|
|
|
|
}
|
|
|
|
|
2016-08-12 20:17:11 +00:00
|
|
|
void SkSVGNode::setStrokeWidth(const SkSVGLength& strokeWidth) {
|
|
|
|
fPresentationAttributes.fStrokeWidth.set(strokeWidth);
|
|
|
|
}
|
|
|
|
|
2017-10-12 15:33:28 +00:00
|
|
|
void SkSVGNode::setVisibility(const SkSVGVisibility& visibility) {
|
2020-04-09 12:54:29 +00:00
|
|
|
SetInheritedByDefault(fPresentationAttributes.fVisibility, visibility);
|
2017-10-12 15:33:28 +00:00
|
|
|
}
|
|
|
|
|
2020-04-10 14:14:04 +00:00
|
|
|
void SkSVGNode::onSetAttribute(SkSVGAttribute attr, const SkSVGValue& v) {
|
2016-07-27 01:46:34 +00:00
|
|
|
switch (attr) {
|
2016-12-08 14:26:47 +00:00
|
|
|
case SkSVGAttribute::kClipPath:
|
2020-04-10 14:14:04 +00:00
|
|
|
if (const SkSVGClipValue* clip = v.as<SkSVGClipValue>()) {
|
2016-12-08 14:26:47 +00:00
|
|
|
this->setClipPath(*clip);
|
|
|
|
}
|
|
|
|
break;
|
2017-10-10 15:22:08 +00:00
|
|
|
case SkSVGAttribute::kClipRule:
|
2020-04-10 14:14:04 +00:00
|
|
|
if (const SkSVGFillRuleValue* clipRule = v.as<SkSVGFillRuleValue>()) {
|
2017-10-10 15:22:08 +00:00
|
|
|
this->setClipRule(*clipRule);
|
|
|
|
}
|
|
|
|
break;
|
2020-04-09 18:14:10 +00:00
|
|
|
case SkSVGAttribute::kColor:
|
2020-04-10 14:14:04 +00:00
|
|
|
if (const SkSVGColorValue* color = v.as<SkSVGColorValue>()) {
|
2020-04-09 18:14:10 +00:00
|
|
|
this->setColor(*color);
|
|
|
|
}
|
|
|
|
break;
|
2016-07-29 15:52:03 +00:00
|
|
|
case SkSVGAttribute::kFill:
|
2020-04-10 14:14:04 +00:00
|
|
|
if (const SkSVGPaintValue* paint = v.as<SkSVGPaintValue>()) {
|
2016-08-12 20:17:11 +00:00
|
|
|
this->setFill(*paint);
|
2016-08-11 16:16:29 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
case SkSVGAttribute::kFillOpacity:
|
2020-04-10 14:14:04 +00:00
|
|
|
if (const SkSVGNumberValue* opacity = v.as<SkSVGNumberValue>()) {
|
2016-08-12 20:17:11 +00:00
|
|
|
this->setFillOpacity(*opacity);
|
2016-07-27 01:46:34 +00:00
|
|
|
}
|
|
|
|
break;
|
2016-12-01 18:35:11 +00:00
|
|
|
case SkSVGAttribute::kFillRule:
|
2020-04-10 14:14:04 +00:00
|
|
|
if (const SkSVGFillRuleValue* fillRule = v.as<SkSVGFillRuleValue>()) {
|
2016-12-01 18:35:11 +00:00
|
|
|
this->setFillRule(*fillRule);
|
|
|
|
}
|
|
|
|
break;
|
2016-08-15 19:45:11 +00:00
|
|
|
case SkSVGAttribute::kOpacity:
|
2020-04-10 14:14:04 +00:00
|
|
|
if (const SkSVGNumberValue* opacity = v.as<SkSVGNumberValue>()) {
|
2016-08-15 19:45:11 +00:00
|
|
|
this->setOpacity(*opacity);
|
|
|
|
}
|
|
|
|
break;
|
2016-07-29 15:52:03 +00:00
|
|
|
case SkSVGAttribute::kStroke:
|
2020-04-10 14:14:04 +00:00
|
|
|
if (const SkSVGPaintValue* paint = v.as<SkSVGPaintValue>()) {
|
2016-08-12 20:17:11 +00:00
|
|
|
this->setStroke(*paint);
|
2016-08-11 16:16:29 +00:00
|
|
|
}
|
|
|
|
break;
|
2017-10-13 18:07:44 +00:00
|
|
|
case SkSVGAttribute::kStrokeDashArray:
|
2020-04-10 14:14:04 +00:00
|
|
|
if (const SkSVGDashArrayValue* dashArray = v.as<SkSVGDashArrayValue>()) {
|
2017-10-13 18:07:44 +00:00
|
|
|
this->setStrokeDashArray(*dashArray);
|
|
|
|
}
|
|
|
|
break;
|
2017-10-13 22:18:32 +00:00
|
|
|
case SkSVGAttribute::kStrokeDashOffset:
|
2020-04-10 14:14:04 +00:00
|
|
|
if (const SkSVGLengthValue* dashOffset= v.as<SkSVGLengthValue>()) {
|
2017-10-13 22:18:32 +00:00
|
|
|
this->setStrokeDashOffset(*dashOffset);
|
|
|
|
}
|
|
|
|
break;
|
2016-08-11 16:16:29 +00:00
|
|
|
case SkSVGAttribute::kStrokeOpacity:
|
2020-04-10 14:14:04 +00:00
|
|
|
if (const SkSVGNumberValue* opacity = v.as<SkSVGNumberValue>()) {
|
2016-08-12 20:17:11 +00:00
|
|
|
this->setStrokeOpacity(*opacity);
|
2016-08-11 16:16:29 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
case SkSVGAttribute::kStrokeLineCap:
|
2020-04-10 14:14:04 +00:00
|
|
|
if (const SkSVGLineCapValue* lineCap = v.as<SkSVGLineCapValue>()) {
|
2020-04-09 12:54:29 +00:00
|
|
|
this->setStrokeLineCap(*lineCap);
|
2016-08-11 16:16:29 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
case SkSVGAttribute::kStrokeLineJoin:
|
2020-04-10 14:14:04 +00:00
|
|
|
if (const SkSVGLineJoinValue* lineJoin = v.as<SkSVGLineJoinValue>()) {
|
2020-04-09 12:54:29 +00:00
|
|
|
this->setStrokeLineJoin(*lineJoin);
|
2016-08-11 16:16:29 +00:00
|
|
|
}
|
|
|
|
break;
|
2017-10-09 16:57:41 +00:00
|
|
|
case SkSVGAttribute::kStrokeMiterLimit:
|
2020-04-10 14:14:04 +00:00
|
|
|
if (const SkSVGNumberValue* miterLimit = v.as<SkSVGNumberValue>()) {
|
2020-04-09 12:54:29 +00:00
|
|
|
this->setStrokeMiterLimit(*miterLimit);
|
2017-10-09 16:57:41 +00:00
|
|
|
}
|
|
|
|
break;
|
2016-08-11 16:16:29 +00:00
|
|
|
case SkSVGAttribute::kStrokeWidth:
|
2020-04-10 14:14:04 +00:00
|
|
|
if (const SkSVGLengthValue* strokeWidth = v.as<SkSVGLengthValue>()) {
|
2016-08-12 20:17:11 +00:00
|
|
|
this->setStrokeWidth(*strokeWidth);
|
2016-07-27 01:46:34 +00:00
|
|
|
}
|
|
|
|
break;
|
2017-10-12 15:33:28 +00:00
|
|
|
case SkSVGAttribute::kVisibility:
|
2020-04-10 14:14:04 +00:00
|
|
|
if (const SkSVGVisibilityValue* visibility = v.as<SkSVGVisibilityValue>()) {
|
2017-10-12 15:33:28 +00:00
|
|
|
this->setVisibility(*visibility);
|
|
|
|
}
|
|
|
|
break;
|
2016-07-27 01:46:34 +00:00
|
|
|
default:
|
2018-02-16 18:49:48 +00:00
|
|
|
#if defined(SK_VERBOSE_SVG_PARSING)
|
|
|
|
SkDebugf("attribute ID <%d> ignored for node <%d>\n", attr, fTag);
|
|
|
|
#endif
|
2016-07-27 01:46:34 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|