summaryrefslogtreecommitdiffstats
path: root/dom/svg/SVGUseElement.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'dom/svg/SVGUseElement.cpp')
-rw-r--r--dom/svg/SVGUseElement.cpp669
1 files changed, 669 insertions, 0 deletions
diff --git a/dom/svg/SVGUseElement.cpp b/dom/svg/SVGUseElement.cpp
new file mode 100644
index 0000000000..3a2bfc63e1
--- /dev/null
+++ b/dom/svg/SVGUseElement.cpp
@@ -0,0 +1,669 @@
+/* -*- 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
+ * file, You can obtain one at http://mozilla.org/MPL/2.0/. */
+
+#include "mozilla/dom/SVGUseElement.h"
+
+#include "mozilla/ArrayUtils.h"
+#include "mozilla/ErrorResult.h"
+#include "mozilla/ScopeExit.h"
+#include "mozilla/StaticPrefs_svg.h"
+#include "mozilla/SVGObserverUtils.h"
+#include "mozilla/SVGUseFrame.h"
+#include "mozilla/URLExtraData.h"
+#include "mozilla/dom/Document.h"
+#include "mozilla/dom/ReferrerInfo.h"
+#include "mozilla/dom/ShadowIncludingTreeIterator.h"
+#include "mozilla/dom/SVGLengthBinding.h"
+#include "mozilla/dom/SVGGraphicsElement.h"
+#include "mozilla/dom/SVGSVGElement.h"
+#include "mozilla/dom/SVGUseElementBinding.h"
+#include "nsGkAtoms.h"
+#include "nsContentUtils.h"
+#include "nsIReferrerInfo.h"
+#include "nsIURI.h"
+#include "SVGGeometryProperty.h"
+
+NS_IMPL_NS_NEW_SVG_ELEMENT(Use)
+
+namespace mozilla::dom {
+
+JSObject* SVGUseElement::WrapNode(JSContext* aCx,
+ JS::Handle<JSObject*> aGivenProto) {
+ return SVGUseElement_Binding::Wrap(aCx, this, aGivenProto);
+}
+
+////////////////////////////////////////////////////////////////////////
+// implementation
+
+SVGElement::LengthInfo SVGUseElement::sLengthInfo[4] = {
+ {nsGkAtoms::x, 0, SVGLength_Binding::SVG_LENGTHTYPE_NUMBER,
+ SVGContentUtils::X},
+ {nsGkAtoms::y, 0, SVGLength_Binding::SVG_LENGTHTYPE_NUMBER,
+ SVGContentUtils::Y},
+ {nsGkAtoms::width, 0, SVGLength_Binding::SVG_LENGTHTYPE_NUMBER,
+ SVGContentUtils::X},
+ {nsGkAtoms::height, 0, SVGLength_Binding::SVG_LENGTHTYPE_NUMBER,
+ SVGContentUtils::Y},
+};
+
+SVGElement::StringInfo SVGUseElement::sStringInfo[2] = {
+ {nsGkAtoms::href, kNameSpaceID_None, true},
+ {nsGkAtoms::href, kNameSpaceID_XLink, true}};
+
+//----------------------------------------------------------------------
+// nsISupports methods
+
+NS_IMPL_CYCLE_COLLECTION_CLASS(SVGUseElement)
+
+NS_IMPL_CYCLE_COLLECTION_UNLINK_BEGIN_INHERITED(SVGUseElement,
+ SVGUseElementBase)
+ nsAutoScriptBlocker scriptBlocker;
+ NS_IMPL_CYCLE_COLLECTION_UNLINK(mOriginal)
+ tmp->UnlinkSource();
+NS_IMPL_CYCLE_COLLECTION_UNLINK_END
+NS_IMPL_CYCLE_COLLECTION_TRAVERSE_BEGIN_INHERITED(SVGUseElement,
+ SVGUseElementBase)
+ NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mOriginal)
+ tmp->mReferencedElementTracker.Traverse(&cb);
+NS_IMPL_CYCLE_COLLECTION_TRAVERSE_END
+
+NS_IMPL_ISUPPORTS_CYCLE_COLLECTION_INHERITED(SVGUseElement, SVGUseElementBase,
+ nsIMutationObserver)
+
+//----------------------------------------------------------------------
+// Implementation
+
+SVGUseElement::SVGUseElement(
+ already_AddRefed<mozilla::dom::NodeInfo>&& aNodeInfo)
+ : SVGUseElementBase(std::move(aNodeInfo)),
+ mReferencedElementTracker(this) {}
+
+SVGUseElement::~SVGUseElement() {
+ UnlinkSource();
+ MOZ_DIAGNOSTIC_ASSERT(!OwnerDoc()->SVGUseElementNeedsShadowTreeUpdate(*this),
+ "Dying without unbinding?");
+}
+
+namespace SVGT = SVGGeometryProperty::Tags;
+
+//----------------------------------------------------------------------
+// nsINode methods
+
+void SVGUseElement::ProcessAttributeChange(int32_t aNamespaceID,
+ nsAtom* aAttribute) {
+ if (aNamespaceID == kNameSpaceID_None) {
+ if (aAttribute == nsGkAtoms::width || aAttribute == nsGkAtoms::height) {
+ const bool hadValidDimensions = HasValidDimensions();
+ const bool isUsed = OurWidthAndHeightAreUsed();
+ if (isUsed) {
+ SyncWidthOrHeight(aAttribute);
+ }
+
+ if (auto* frame = GetFrame()) {
+ frame->DimensionAttributeChanged(hadValidDimensions, isUsed);
+ }
+ }
+ }
+
+ if ((aNamespaceID == kNameSpaceID_XLink ||
+ aNamespaceID == kNameSpaceID_None) &&
+ aAttribute == nsGkAtoms::href) {
+ // We're changing our nature, clear out the clone information.
+ if (auto* frame = GetFrame()) {
+ frame->HrefChanged();
+ }
+ mOriginal = nullptr;
+ UnlinkSource();
+ TriggerReclone();
+ }
+}
+
+void SVGUseElement::AfterSetAttr(int32_t aNamespaceID, nsAtom* aAttribute,
+ const nsAttrValue* aValue,
+ const nsAttrValue* aOldValue,
+ nsIPrincipal* aSubjectPrincipal,
+ bool aNotify) {
+ ProcessAttributeChange(aNamespaceID, aAttribute);
+ return SVGUseElementBase::AfterSetAttr(aNamespaceID, aAttribute, aValue,
+ aOldValue, aSubjectPrincipal, aNotify);
+}
+
+nsresult SVGUseElement::Clone(dom::NodeInfo* aNodeInfo,
+ nsINode** aResult) const {
+ *aResult = nullptr;
+ SVGUseElement* it =
+ new (aNodeInfo->NodeInfoManager()) SVGUseElement(do_AddRef(aNodeInfo));
+
+ nsCOMPtr<nsINode> kungFuDeathGrip(it);
+ nsresult rv1 = it->Init();
+ nsresult rv2 = const_cast<SVGUseElement*>(this)->CopyInnerTo(it);
+
+ // SVGUseElement specific portion - record who we cloned from
+ it->mOriginal = const_cast<SVGUseElement*>(this);
+
+ if (NS_SUCCEEDED(rv1) && NS_SUCCEEDED(rv2)) {
+ kungFuDeathGrip.swap(*aResult);
+ }
+
+ return NS_FAILED(rv1) ? rv1 : rv2;
+}
+
+nsresult SVGUseElement::BindToTree(BindContext& aContext, nsINode& aParent) {
+ nsresult rv = SVGUseElementBase::BindToTree(aContext, aParent);
+ NS_ENSURE_SUCCESS(rv, rv);
+
+ TriggerReclone();
+ return NS_OK;
+}
+
+void SVGUseElement::UnbindFromTree(bool aNullParent) {
+ SVGUseElementBase::UnbindFromTree(aNullParent);
+ OwnerDoc()->UnscheduleSVGUseElementShadowTreeUpdate(*this);
+}
+
+already_AddRefed<DOMSVGAnimatedString> SVGUseElement::Href() {
+ return mStringAttributes[HREF].IsExplicitlySet()
+ ? mStringAttributes[HREF].ToDOMAnimatedString(this)
+ : mStringAttributes[XLINK_HREF].ToDOMAnimatedString(this);
+}
+
+//----------------------------------------------------------------------
+
+already_AddRefed<DOMSVGAnimatedLength> SVGUseElement::X() {
+ return mLengthAttributes[ATTR_X].ToDOMAnimatedLength(this);
+}
+
+already_AddRefed<DOMSVGAnimatedLength> SVGUseElement::Y() {
+ return mLengthAttributes[ATTR_Y].ToDOMAnimatedLength(this);
+}
+
+already_AddRefed<DOMSVGAnimatedLength> SVGUseElement::Width() {
+ return mLengthAttributes[ATTR_WIDTH].ToDOMAnimatedLength(this);
+}
+
+already_AddRefed<DOMSVGAnimatedLength> SVGUseElement::Height() {
+ return mLengthAttributes[ATTR_HEIGHT].ToDOMAnimatedLength(this);
+}
+
+//----------------------------------------------------------------------
+// nsIMutationObserver methods
+
+void SVGUseElement::CharacterDataChanged(nsIContent* aContent,
+ const CharacterDataChangeInfo&) {
+ if (nsContentUtils::IsInSameAnonymousTree(mReferencedElementTracker.get(),
+ aContent)) {
+ TriggerReclone();
+ }
+}
+
+void SVGUseElement::AttributeChanged(Element* aElement, int32_t aNamespaceID,
+ nsAtom* aAttribute, int32_t aModType,
+ const nsAttrValue* aOldValue) {
+ if (nsContentUtils::IsInSameAnonymousTree(mReferencedElementTracker.get(),
+ aElement)) {
+ TriggerReclone();
+ }
+}
+
+void SVGUseElement::ContentAppended(nsIContent* aFirstNewContent) {
+ // FIXME(emilio, bug 1442336): Why does this check the parent but
+ // ContentInserted the child?
+ if (nsContentUtils::IsInSameAnonymousTree(mReferencedElementTracker.get(),
+ aFirstNewContent->GetParent())) {
+ TriggerReclone();
+ }
+}
+
+void SVGUseElement::ContentInserted(nsIContent* aChild) {
+ // FIXME(emilio, bug 1442336): Why does this check the child but
+ // ContentAppended the parent?
+ if (nsContentUtils::IsInSameAnonymousTree(mReferencedElementTracker.get(),
+ aChild)) {
+ TriggerReclone();
+ }
+}
+
+void SVGUseElement::ContentRemoved(nsIContent* aChild,
+ nsIContent* aPreviousSibling) {
+ if (nsContentUtils::IsInSameAnonymousTree(mReferencedElementTracker.get(),
+ aChild)) {
+ TriggerReclone();
+ }
+}
+
+void SVGUseElement::NodeWillBeDestroyed(nsINode* aNode) {
+ nsCOMPtr<nsIMutationObserver> kungFuDeathGrip(this);
+ UnlinkSource();
+}
+
+// Returns whether this node could ever be displayed.
+static bool NodeCouldBeRendered(const nsINode& aNode) {
+ if (aNode.IsSVGElement(nsGkAtoms::symbol)) {
+ // Only <symbol> elements in the root of a <svg:use> shadow tree are
+ // displayed.
+ auto* shadowRoot = ShadowRoot::FromNodeOrNull(aNode.GetParentNode());
+ return shadowRoot && shadowRoot->Host()->IsSVGElement(nsGkAtoms::use);
+ }
+ // TODO: Do we have other cases we can optimize out easily?
+ return true;
+}
+
+// <svg:use> can be used (no pun intended) to trivially cause an explosion of
+// clones that could potentially DoS the browser. We have a configurable limit
+// to control this.
+static bool IsTooMuchRecursion(uint32_t aCount) {
+ switch (StaticPrefs::svg_use_element_recursive_clone_limit_enabled()) {
+ case 0:
+ return false;
+ case 1:
+ break;
+ default:
+ if (!XRE_IsParentProcess()) {
+ return false;
+ }
+ break;
+ }
+ return aCount >= StaticPrefs::svg_use_element_recursive_clone_limit();
+}
+
+// Circular loop detection, plus detection of whether this shadow tree is
+// rendered at all.
+auto SVGUseElement::ScanAncestors(const Element& aTarget) const -> ScanResult {
+ uint32_t count = 0;
+ return ScanAncestorsInternal(aTarget, count);
+}
+
+auto SVGUseElement::ScanAncestorsInternal(const Element& aTarget,
+ uint32_t& aCount) const
+ -> ScanResult {
+ if (&aTarget == this) {
+ return ScanResult::CyclicReference;
+ }
+ if (mOriginal) {
+ if (IsTooMuchRecursion(++aCount)) {
+ return ScanResult::TooDeep;
+ }
+ auto result = mOriginal->ScanAncestorsInternal(aTarget, aCount);
+ switch (result) {
+ case ScanResult::TooDeep:
+ case ScanResult::CyclicReference:
+ return result;
+ case ScanResult::Ok:
+ case ScanResult::Invisible:
+ break;
+ }
+ }
+
+ auto result = ScanResult::Ok;
+ for (nsINode* parent = GetParentOrShadowHostNode(); parent;
+ parent = parent->GetParentOrShadowHostNode()) {
+ if (parent == &aTarget) {
+ return ScanResult::CyclicReference;
+ }
+ if (auto* use = SVGUseElement::FromNode(*parent)) {
+ if (IsTooMuchRecursion(++aCount)) {
+ return ScanResult::TooDeep;
+ }
+ if (mOriginal && use->mOriginal == mOriginal) {
+ return ScanResult::CyclicReference;
+ }
+ }
+ // Do we have other similar cases we can optimize out easily?
+ if (!NodeCouldBeRendered(*parent)) {
+ // NOTE(emilio): We can't just return here. If we're cyclic, we need to
+ // know.
+ result = ScanResult::Invisible;
+ }
+ }
+ return result;
+}
+
+//----------------------------------------------------------------------
+
+static bool IsForbiddenUseNode(const nsINode& aNode) {
+ if (!aNode.IsElement()) {
+ return false;
+ }
+ const auto* svg = SVGElement::FromNode(aNode);
+ return !svg || !svg->IsSVGGraphicsElement();
+}
+
+static void CollectForbiddenNodes(Element& aRoot,
+ nsTArray<RefPtr<nsINode>>& aNodes) {
+ auto iter = dom::ShadowIncludingTreeIterator(aRoot);
+ while (iter) {
+ nsINode* node = *iter;
+ if (IsForbiddenUseNode(*node)) {
+ aNodes.AppendElement(node);
+ iter.SkipChildren();
+ continue;
+ }
+ ++iter;
+ }
+}
+
+// SVG1 restricted <use> trees to SVGGraphicsElements.
+// https://www.w3.org/TR/SVG11/struct.html#UseElement:
+//
+// Any ‘svg’, ‘symbol’, ‘g’, graphics element or other ‘use’ is potentially a
+// template object that can be re-used (i.e., "instanced") in the SVG
+// document via a ‘use’ element. The ‘use’ element references another element
+// and indicates that the graphical contents of that element is
+// included/drawn at that given point in the document.
+//
+// SVG2 doesn't have that same restriction.
+// https://www.w3.org/TR/SVG2/struct.html#UseShadowTree:
+//
+// Previous versions of SVG restricted the contents of the shadow tree to SVG
+// graphics elements. This specification allows any valid SVG document
+// subtree to be cloned. Cloning non-graphical content, however, will not
+// usually have any visible effect.
+//
+// But it's pretty ambiguous as to what the behavior should be for some
+// elements, because <script> is inert, but <iframe> is not, see:
+// https://github.com/w3c/svgwg/issues/876
+//
+// So, fairly confusing, all-in-all.
+static void RemoveForbiddenNodes(Element& aRoot, bool aIsCrossDocument) {
+ switch (StaticPrefs::svg_use_element_graphics_element_restrictions()) {
+ case 0:
+ return;
+ case 1:
+ if (!aIsCrossDocument) {
+ return;
+ }
+ break;
+ default:
+ break;
+ }
+
+ AutoTArray<RefPtr<nsINode>, 10> unsafeNodes;
+ CollectForbiddenNodes(aRoot, unsafeNodes);
+ for (auto& unsafeNode : unsafeNodes) {
+ unsafeNode->Remove();
+ }
+}
+
+void SVGUseElement::UpdateShadowTree() {
+ MOZ_ASSERT(IsInComposedDoc());
+
+ if (mReferencedElementTracker.get()) {
+ mReferencedElementTracker.get()->RemoveMutationObserver(this);
+ }
+
+ LookupHref();
+
+ RefPtr<ShadowRoot> shadow = GetShadowRoot();
+ if (!shadow) {
+ shadow = AttachShadowWithoutNameChecks(ShadowRootMode::Closed);
+ }
+ MOZ_ASSERT(shadow);
+
+ auto* targetElement =
+ SVGGraphicsElement::FromNodeOrNull(mReferencedElementTracker.get());
+ RefPtr<Element> newElement;
+
+ auto UpdateShadowTree = mozilla::MakeScopeExit([&]() {
+ nsIContent* firstChild = shadow->GetFirstChild();
+ if (firstChild) {
+ MOZ_ASSERT(!firstChild->GetNextSibling());
+ shadow->RemoveChildNode(firstChild, /* aNotify = */ true);
+ }
+
+ if (newElement) {
+ shadow->AppendChildTo(newElement, /* aNotify = */ true, IgnoreErrors());
+ }
+ });
+
+ // make sure target is valid type for <use>
+ if (!targetElement) {
+ return;
+ }
+
+ if (ScanAncestors(*targetElement) != ScanResult::Ok) {
+ return;
+ }
+
+ nsCOMPtr<nsIURI> baseURI = targetElement->GetBaseURI();
+ if (!baseURI) {
+ return;
+ }
+
+ {
+ const bool isCrossDocument = targetElement->OwnerDoc() != OwnerDoc();
+
+ nsNodeInfoManager* nodeInfoManager =
+ isCrossDocument ? OwnerDoc()->NodeInfoManager() : nullptr;
+
+ nsCOMPtr<nsINode> newNode =
+ targetElement->Clone(true, nodeInfoManager, IgnoreErrors());
+ if (!newNode) {
+ return;
+ }
+
+ MOZ_ASSERT(newNode->IsElement());
+ newElement = newNode.forget().downcast<Element>();
+ RemoveForbiddenNodes(*newElement, isCrossDocument);
+ }
+
+ if (newElement->IsAnyOfSVGElements(nsGkAtoms::svg, nsGkAtoms::symbol)) {
+ auto* newSVGElement = static_cast<SVGElement*>(newElement.get());
+ if (mLengthAttributes[ATTR_WIDTH].IsExplicitlySet())
+ newSVGElement->SetLength(nsGkAtoms::width, mLengthAttributes[ATTR_WIDTH]);
+ if (mLengthAttributes[ATTR_HEIGHT].IsExplicitlySet())
+ newSVGElement->SetLength(nsGkAtoms::height,
+ mLengthAttributes[ATTR_HEIGHT]);
+ }
+
+ // Bug 1415044 the specs do not say which referrer information we should use.
+ // This may change if there's any spec comes out.
+ auto referrerInfo = MakeRefPtr<ReferrerInfo>(*this);
+ mContentURLData = new URLExtraData(baseURI.forget(), referrerInfo.forget(),
+ do_AddRef(NodePrincipal()));
+
+ targetElement->AddMutationObserver(this);
+}
+
+nsIURI* SVGUseElement::GetSourceDocURI() {
+ nsIContent* targetElement = mReferencedElementTracker.get();
+ if (!targetElement) {
+ return nullptr;
+ }
+
+ return targetElement->OwnerDoc()->GetDocumentURI();
+}
+
+const Encoding* SVGUseElement::GetSourceDocCharacterSet() {
+ nsIContent* targetElement = mReferencedElementTracker.get();
+ if (!targetElement) {
+ return nullptr;
+ }
+
+ return targetElement->OwnerDoc()->GetDocumentCharacterSet();
+}
+
+static nsINode* GetClonedChild(const SVGUseElement& aUseElement) {
+ const ShadowRoot* shadow = aUseElement.GetShadowRoot();
+ return shadow ? shadow->GetFirstChild() : nullptr;
+}
+
+bool SVGUseElement::OurWidthAndHeightAreUsed() const {
+ nsINode* clonedChild = GetClonedChild(*this);
+ return clonedChild &&
+ clonedChild->IsAnyOfSVGElements(nsGkAtoms::svg, nsGkAtoms::symbol);
+}
+
+//----------------------------------------------------------------------
+// implementation helpers
+
+void SVGUseElement::SyncWidthOrHeight(nsAtom* aName) {
+ NS_ASSERTION(aName == nsGkAtoms::width || aName == nsGkAtoms::height,
+ "The clue is in the function name");
+ NS_ASSERTION(OurWidthAndHeightAreUsed(), "Don't call this");
+
+ if (!OurWidthAndHeightAreUsed()) {
+ return;
+ }
+
+ auto* target = SVGElement::FromNode(GetClonedChild(*this));
+ uint32_t index =
+ sLengthInfo[ATTR_WIDTH].mName == aName ? ATTR_WIDTH : ATTR_HEIGHT;
+
+ if (mLengthAttributes[index].IsExplicitlySet()) {
+ target->SetLength(aName, mLengthAttributes[index]);
+ return;
+ }
+ if (target->IsSVGElement(nsGkAtoms::svg)) {
+ // Our width/height attribute is now no longer explicitly set, so we
+ // need to revert the clone's width/height to the width/height of the
+ // content that's being cloned.
+ TriggerReclone();
+ return;
+ }
+ // Our width/height attribute is now no longer explicitly set, so we
+ // need to set the value to 100%
+ SVGAnimatedLength length;
+ length.Init(SVGContentUtils::XY, 0xff, 100,
+ SVGLength_Binding::SVG_LENGTHTYPE_PERCENTAGE);
+ target->SetLength(aName, length);
+}
+
+void SVGUseElement::LookupHref() {
+ nsAutoString href;
+ if (mStringAttributes[HREF].IsExplicitlySet()) {
+ mStringAttributes[HREF].GetAnimValue(href, this);
+ } else {
+ mStringAttributes[XLINK_HREF].GetAnimValue(href, this);
+ }
+
+ if (href.IsEmpty()) {
+ return;
+ }
+
+ nsCOMPtr<nsIURI> originURI =
+ mOriginal ? mOriginal->GetBaseURI() : GetBaseURI();
+ nsCOMPtr<nsIURI> baseURI =
+ nsContentUtils::IsLocalRefURL(href)
+ ? SVGObserverUtils::GetBaseURLForLocalRef(this, originURI)
+ : originURI;
+
+ nsCOMPtr<nsIURI> targetURI;
+ nsContentUtils::NewURIWithDocumentCharset(getter_AddRefs(targetURI), href,
+ GetComposedDoc(), baseURI);
+ nsIReferrerInfo* referrer =
+ OwnerDoc()->ReferrerInfoForInternalCSSAndSVGResources();
+ mReferencedElementTracker.ResetToURIFragmentID(this, targetURI, referrer);
+}
+
+void SVGUseElement::TriggerReclone() {
+ if (Document* doc = GetComposedDoc()) {
+ doc->ScheduleSVGUseElementShadowTreeUpdate(*this);
+ }
+}
+
+void SVGUseElement::UnlinkSource() {
+ if (mReferencedElementTracker.get()) {
+ mReferencedElementTracker.get()->RemoveMutationObserver(this);
+ }
+ mReferencedElementTracker.Unlink();
+}
+
+//----------------------------------------------------------------------
+// SVGElement methods
+
+/* virtual */
+gfxMatrix SVGUseElement::PrependLocalTransformsTo(
+ const gfxMatrix& aMatrix, SVGTransformTypes aWhich) const {
+ // 'transform' attribute:
+ gfxMatrix userToParent;
+
+ if (aWhich == eUserSpaceToParent || aWhich == eAllTransforms) {
+ userToParent = GetUserToParentTransform(mAnimateMotionTransform.get(),
+ mTransforms.get());
+ if (aWhich == eUserSpaceToParent) {
+ return userToParent * aMatrix;
+ }
+ }
+
+ // our 'x' and 'y' attributes:
+ float x, y;
+ if (!SVGGeometryProperty::ResolveAll<SVGT::X, SVGT::Y>(this, &x, &y)) {
+ const_cast<SVGUseElement*>(this)->GetAnimatedLengthValues(&x, &y, nullptr);
+ }
+
+ gfxMatrix childToUser = gfxMatrix::Translation(x, y);
+
+ if (aWhich == eAllTransforms) {
+ return childToUser * userToParent * aMatrix;
+ }
+
+ MOZ_ASSERT(aWhich == eChildToUserSpace, "Unknown TransformTypes");
+
+ // The following may look broken because pre-multiplying our eChildToUserSpace
+ // transform with another matrix without including our eUserSpaceToParent
+ // transform between the two wouldn't make sense. We don't expect that to
+ // ever happen though. We get here either when the identity matrix has been
+ // passed because our caller just wants our eChildToUserSpace transform, or
+ // when our eUserSpaceToParent transform has already been multiplied into the
+ // matrix that our caller passes (such as when we're called from PaintSVG).
+ return childToUser * aMatrix;
+}
+
+/* virtual */
+bool SVGUseElement::HasValidDimensions() const {
+ if (!OurWidthAndHeightAreUsed()) {
+ return true;
+ }
+
+ return (!mLengthAttributes[ATTR_WIDTH].IsExplicitlySet() ||
+ mLengthAttributes[ATTR_WIDTH].GetAnimValInSpecifiedUnits() > 0) &&
+ (!mLengthAttributes[ATTR_HEIGHT].IsExplicitlySet() ||
+ mLengthAttributes[ATTR_HEIGHT].GetAnimValInSpecifiedUnits() > 0);
+}
+
+SVGElement::LengthAttributesInfo SVGUseElement::GetLengthInfo() {
+ return LengthAttributesInfo(mLengthAttributes, sLengthInfo,
+ ArrayLength(sLengthInfo));
+}
+
+SVGElement::StringAttributesInfo SVGUseElement::GetStringInfo() {
+ return StringAttributesInfo(mStringAttributes, sStringInfo,
+ ArrayLength(sStringInfo));
+}
+
+SVGUseFrame* SVGUseElement::GetFrame() const {
+ nsIFrame* frame = GetPrimaryFrame();
+ // We might be a plain SVGContainerFrame if we didn't pass the conditional
+ // processing checks.
+ if (!frame || !frame->IsSVGUseFrame()) {
+ MOZ_ASSERT_IF(frame, frame->Type() == LayoutFrameType::None);
+ return nullptr;
+ }
+ return static_cast<SVGUseFrame*>(frame);
+}
+
+//----------------------------------------------------------------------
+// nsIContent methods
+
+NS_IMETHODIMP_(bool)
+SVGUseElement::IsAttributeMapped(const nsAtom* name) const {
+ return name == nsGkAtoms::x || name == nsGkAtoms::y ||
+ SVGUseElementBase::IsAttributeMapped(name);
+}
+
+nsCSSPropertyID SVGUseElement::GetCSSPropertyIdForAttrEnum(uint8_t aAttrEnum) {
+ switch (aAttrEnum) {
+ case ATTR_X:
+ return eCSSProperty_x;
+ case ATTR_Y:
+ return eCSSProperty_y;
+ default:
+ // Currently we don't map width or height to style
+ return eCSSProperty_UNKNOWN;
+ }
+}
+
+} // namespace mozilla::dom