Source code
Revision control
Copy as Markdown
Other Tools
/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
/* vim: set ts=8 sts=2 et sw=2 tw=80: */
/* This Source Code Form is subject to the terms of the Mozilla Public
* License, v. 2.0. If a copy of the MPL was not distributed with this
#include "mozilla/dom/SVGPathSegment.h"
#include "mozilla/dom/SVGPathElementBinding.h"
#include "SVGPathSegUtils.h"
namespace mozilla::dom {
JSObject* SVGPathSegment::WrapObject(JSContext* aCx,
JS::Handle<JSObject*> aGivenProto) {
return SVGPathSegment_Binding::Wrap(aCx, this, aGivenProto);
}
NS_IMPL_CYCLE_COLLECTION_WRAPPERCACHE(SVGPathSegment, mSVGPathElement)
//----------------------------------------------------------------------
// Implementation
SVGPathSegment::SVGPathSegment(SVGPathElement* aSVGPathElement,
const StylePathCommand& aCommand)
: mSVGPathElement(aSVGPathElement) {
switch (aCommand.tag) {
case StylePathCommand::Tag::Close:
mCommand.AssignLiteral("Z");
break;
case StylePathCommand::Tag::Move:
mCommand.AssignLiteral(aCommand.move.by_to == StyleByTo::To ? "M" : "m");
mValues.AppendElement(aCommand.move.point.x);
mValues.AppendElement(aCommand.move.point.y);
break;
case StylePathCommand::Tag::Line:
mCommand.AssignLiteral(aCommand.line.by_to == StyleByTo::To ? "L" : "l");
mValues.AppendElement(aCommand.line.point.x);
mValues.AppendElement(aCommand.line.point.y);
break;
case StylePathCommand::Tag::CubicCurve:
mCommand.AssignLiteral(aCommand.cubic_curve.by_to == StyleByTo::To ? "C"
: "c");
mValues.AppendElement(aCommand.cubic_curve.control1.x);
mValues.AppendElement(aCommand.cubic_curve.control1.y);
mValues.AppendElement(aCommand.cubic_curve.control2.x);
mValues.AppendElement(aCommand.cubic_curve.control2.y);
mValues.AppendElement(aCommand.cubic_curve.point.x);
mValues.AppendElement(aCommand.cubic_curve.point.y);
break;
case StylePathCommand::Tag::QuadCurve:
mCommand.AssignLiteral(aCommand.quad_curve.by_to == StyleByTo::To ? "Q"
: "q");
mValues.AppendElement(aCommand.quad_curve.control1.x);
mValues.AppendElement(aCommand.quad_curve.control1.y);
mValues.AppendElement(aCommand.quad_curve.point.x);
mValues.AppendElement(aCommand.quad_curve.point.y);
break;
case StylePathCommand::Tag::Arc:
mCommand.AssignLiteral(aCommand.arc.by_to == StyleByTo::To ? "A" : "a");
mValues.AppendElement(aCommand.arc.radii.x);
mValues.AppendElement(aCommand.arc.radii.y);
mValues.AppendElement(aCommand.arc.rotate);
mValues.AppendElement(aCommand.arc.arc_size == StyleArcSize::Large);
mValues.AppendElement(aCommand.arc.arc_sweep == StyleArcSweep::Cw);
mValues.AppendElement(aCommand.arc.point.x);
mValues.AppendElement(aCommand.arc.point.y);
break;
case StylePathCommand::Tag::HLine:
mCommand.AssignLiteral(aCommand.h_line.by_to == StyleByTo::To ? "H"
: "h");
mValues.AppendElement(aCommand.h_line.x);
break;
case StylePathCommand::Tag::VLine:
mCommand.AssignLiteral(aCommand.v_line.by_to == StyleByTo::To ? "V"
: "v");
mValues.AppendElement(aCommand.v_line.y);
break;
case StylePathCommand::Tag::SmoothCubic:
mCommand.AssignLiteral(
aCommand.smooth_cubic.by_to == StyleByTo::To ? "S" : "s");
mValues.AppendElement(aCommand.smooth_cubic.control2.x);
mValues.AppendElement(aCommand.smooth_cubic.control2.y);
mValues.AppendElement(aCommand.smooth_cubic.point.x);
mValues.AppendElement(aCommand.smooth_cubic.point.y);
break;
case StylePathCommand::Tag::SmoothQuad:
mCommand.AssignLiteral(aCommand.smooth_quad.by_to == StyleByTo::To ? "T"
: "t");
mValues.AppendElement(aCommand.smooth_quad.point.x);
mValues.AppendElement(aCommand.smooth_quad.point.y);
break;
}
}
static int32_t ArgCountForType(char aType) {
switch (ToLowerCase(aType)) {
case 'z':
return 0;
case 'm':
case 'l':
return 2;
case 'c':
return 6;
case 'q':
return 4;
case 'a':
return 7;
case 'h':
case 'v':
return 1;
case 's':
return 4;
case 't':
return 2;
}
return -1;
}
bool SVGPathSegment::IsMove() const {
return mCommand.EqualsLiteral("M") || mCommand.EqualsLiteral("m");
}
bool SVGPathSegment::IsArc() const {
return mCommand.EqualsLiteral("A") || mCommand.EqualsLiteral("a");
}
bool IsValidFlag(float aFlag) { return aFlag == 0.0f || aFlag == 1.0f; }
bool SVGPathSegment::IsValid() const {
if (mCommand.Length() != 1) {
return false;
}
auto expectedArgCount = ArgCountForType(mCommand.First());
if (expectedArgCount < 0 || mValues.Length() != uint32_t(expectedArgCount)) {
return false;
}
if (IsArc() && !(IsValidFlag(mValues[3]) && IsValidFlag(mValues[4]))) {
return false;
}
return true;
}
StylePathCommand SVGPathSegment::ToStylePathCommand() const {
MOZ_ASSERT(IsValid(), "Trying to convert invalid SVGPathSegment");
switch (mCommand.First()) {
case 'M':
return StylePathCommand::Move(StyleByTo::To, {mValues[0], mValues[1]});
case 'm':
return StylePathCommand::Move(StyleByTo::By, {mValues[0], mValues[1]});
case 'L':
return StylePathCommand::Line(StyleByTo::To, {mValues[0], mValues[1]});
case 'l':
return StylePathCommand::Line(StyleByTo::By, {mValues[0], mValues[1]});
case 'C':
return StylePathCommand::CubicCurve(
StyleByTo::To, {mValues[4], mValues[5]}, {mValues[0], mValues[1]},
{mValues[2], mValues[3]});
case 'c':
return StylePathCommand::CubicCurve(
StyleByTo::By, {mValues[4], mValues[5]}, {mValues[0], mValues[1]},
{mValues[2], mValues[3]});
case 'Q':
return StylePathCommand::QuadCurve(
StyleByTo::To, {mValues[2], mValues[3]}, {mValues[0], mValues[1]});
case 'q':
return StylePathCommand::QuadCurve(
StyleByTo::By, {mValues[2], mValues[3]}, {mValues[0], mValues[1]});
case 'A':
return StylePathCommand::Arc(
StyleByTo::To, {mValues[5], mValues[6]}, {mValues[0], mValues[1]},
mValues[3] ? StyleArcSweep::Cw : StyleArcSweep::Ccw,
mValues[4] ? StyleArcSize::Large : StyleArcSize::Small, mValues[2]);
case 'a':
return StylePathCommand::Arc(
StyleByTo::By, {mValues[5], mValues[6]}, {mValues[0], mValues[1]},
mValues[3] ? StyleArcSweep::Cw : StyleArcSweep::Ccw,
mValues[4] ? StyleArcSize::Large : StyleArcSize::Small, mValues[2]);
case 'H':
return StylePathCommand::HLine(StyleByTo::To, mValues[0]);
case 'h':
return StylePathCommand::HLine(StyleByTo::By, mValues[0]);
case 'V':
return StylePathCommand::VLine(StyleByTo::To, mValues[0]);
case 'v':
return StylePathCommand::VLine(StyleByTo::By, mValues[0]);
case 'S':
return StylePathCommand::SmoothCubic(
StyleByTo::To, {mValues[2], mValues[3]}, {mValues[0], mValues[1]});
case 's':
return StylePathCommand::SmoothCubic(
StyleByTo::By, {mValues[2], mValues[3]}, {mValues[0], mValues[1]});
case 'T':
return StylePathCommand::SmoothQuad(StyleByTo::To,
{mValues[0], mValues[1]});
case 't':
return StylePathCommand::SmoothQuad(StyleByTo::By,
{mValues[0], mValues[1]});
}
return StylePathCommand::Close();
}
void SVGPathSegment::GetType(DOMString& aType) {
aType.SetKnownLiveString(mCommand);
}
void SVGPathSegment::SetType(const nsAString& aType) { mCommand = aType; }
void SVGPathSegment::GetValues(nsTArray<float>& aValues) {
aValues = mValues.Clone();
}
void SVGPathSegment::SetValues(const nsTArray<float>& aValues) {
mValues = aValues.Clone();
}
} // namespace mozilla::dom