|
|
/*
|
|
|
* Copyright 2016 Google Inc.
|
|
|
*
|
|
|
* Use of this source code is governed by a BSD-style license that can be
|
|
|
* found in the LICENSE file.
|
|
|
*/
|
|
|
|
|
|
#include <tuple>
|
|
|
|
|
|
#include "include/core/SkCanvas.h"
|
|
|
#include "include/core/SkRect.h"
|
|
|
#include "modules/svg/include/SkSVGRect.h"
|
|
|
#include "modules/svg/include/SkSVGRenderContext.h"
|
|
|
#include "modules/svg/include/SkSVGValue.h"
|
|
|
|
|
|
SkSVGRect::SkSVGRect() : INHERITED(SkSVGTag::kRect) {}
|
|
|
|
|
|
bool SkSVGRect::parseAndSetAttribute(const char* n, const char* v) {
|
|
|
return INHERITED::parseAndSetAttribute(n, v) ||
|
|
|
this->setX(SkSVGAttributeParser::parse<SkSVGLength>("x", n, v)) ||
|
|
|
this->setY(SkSVGAttributeParser::parse<SkSVGLength>("y", n, v)) ||
|
|
|
this->setWidth(SkSVGAttributeParser::parse<SkSVGLength>("width", n, v)) ||
|
|
|
this->setHeight(SkSVGAttributeParser::parse<SkSVGLength>("height", n, v)) ||
|
|
|
this->setRx(SkSVGAttributeParser::parse<SkSVGLength>("rx", n, v)) ||
|
|
|
this->setRy(SkSVGAttributeParser::parse<SkSVGLength>("ry", n, v));
|
|
|
}
|
|
|
|
|
|
SkRRect SkSVGRect::resolve(const SkSVGLengthContext& lctx) const {
|
|
|
const auto rect = lctx.resolveRect(fX, fY, fWidth, fHeight);
|
|
|
|
|
|
// https://www.w3.org/TR/SVG11/shapes.html#RectElementRXAttribute:
|
|
|
//
|
|
|
// - Let rx and ry be length values.
|
|
|
// - If neither ‘rx’ nor ‘ry’ are properly specified, then set both rx and ry to 0.
|
|
|
// - Otherwise, if a properly specified value is provided for ‘rx’, but not for ‘ry’,
|
|
|
// then set both rx and ry to the value of ‘rx’.
|
|
|
// - Otherwise, if a properly specified value is provided for ‘ry’, but not for ‘rx’,
|
|
|
// then set both rx and ry to the value of ‘ry’.
|
|
|
// - Otherwise, both ‘rx’ and ‘ry’ were specified properly. Set rx to the value of ‘rx’
|
|
|
// and ry to the value of ‘ry’.
|
|
|
// - If rx is greater than half of ‘width’, then set rx to half of ‘width’.
|
|
|
// - If ry is greater than half of ‘height’, then set ry to half of ‘height’.
|
|
|
// - The effective values of ‘rx’ and ‘ry’ are rx and ry, respectively.
|
|
|
//
|
|
|
auto radii = [this]() {
|
|
|
return fRx.isValid()
|
|
|
? fRy.isValid()
|
|
|
? std::make_tuple(*fRx, *fRy)
|
|
|
: std::make_tuple(*fRx, *fRx)
|
|
|
: fRy.isValid()
|
|
|
? std::make_tuple(*fRy, *fRy)
|
|
|
: std::make_tuple(SkSVGLength(0), SkSVGLength(0));
|
|
|
};
|
|
|
|
|
|
const auto [ rxlen, rylen ] = radii();
|
|
|
const auto rx = std::min(lctx.resolve(rxlen, SkSVGLengthContext::LengthType::kHorizontal),
|
|
|
rect.width() / 2),
|
|
|
ry = std::min(lctx.resolve(rylen, SkSVGLengthContext::LengthType::kVertical),
|
|
|
rect.height() / 2);
|
|
|
|
|
|
return SkRRect::MakeRectXY(rect, rx, ry);
|
|
|
}
|
|
|
|
|
|
void SkSVGRect::onDraw(SkCanvas* canvas, const SkSVGLengthContext& lctx,
|
|
|
const SkPaint& paint, SkPathFillType) const {
|
|
|
canvas->drawRRect(this->resolve(lctx), paint);
|
|
|
}
|
|
|
|
|
|
SkPath SkSVGRect::onAsPath(const SkSVGRenderContext& ctx) const {
|
|
|
SkPath path = SkPath::RRect(this->resolve(ctx.lengthContext()));
|
|
|
|
|
|
this->mapToParent(&path);
|
|
|
|
|
|
return path;
|
|
|
}
|
|
|
|
|
|
SkRect SkSVGRect::onObjectBoundingBox(const SkSVGRenderContext& ctx) const {
|
|
|
return ctx.lengthContext().resolveRect(fX, fY, fWidth, fHeight);
|
|
|
}
|