1 /* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
2 /* vim: set ts=8 sts=2 et sw=2 tw=80: */
3 /* This Source Code Form is subject to the terms of the Mozilla Public
4 * License, v. 2.0. If a copy of the MPL was not distributed with this
5 * file, You can obtain one at http://mozilla.org/MPL/2.0/. */
7 #include "mozilla/dom/SVGGraphicsElement.h"
9 #include "mozilla/dom/BindContext.h"
10 #include "mozilla/dom/Document.h"
11 #include "mozilla/dom/SVGGraphicsElementBinding.h"
12 #include "mozilla/dom/SVGMatrix.h"
13 #include "mozilla/dom/SVGRect.h"
14 #include "mozilla/dom/SVGSVGElement.h"
15 #include "mozilla/ISVGDisplayableFrame.h"
16 #include "mozilla/SVGContentUtils.h"
17 #include "mozilla/SVGTextFrame.h"
18 #include "mozilla/SVGUtils.h"
20 #include "nsIContentInlines.h"
21 #include "nsLayoutUtils.h"
23 namespace mozilla::dom
{
25 //----------------------------------------------------------------------
26 // nsISupports methods
28 NS_IMPL_ADDREF_INHERITED(SVGGraphicsElement
, SVGGraphicsElementBase
)
29 NS_IMPL_RELEASE_INHERITED(SVGGraphicsElement
, SVGGraphicsElementBase
)
31 NS_INTERFACE_MAP_BEGIN(SVGGraphicsElement
)
32 NS_INTERFACE_MAP_ENTRY(mozilla::dom::SVGTests
)
33 NS_INTERFACE_MAP_END_INHERITING(SVGGraphicsElementBase
)
35 //----------------------------------------------------------------------
38 SVGGraphicsElement::SVGGraphicsElement(
39 already_AddRefed
<mozilla::dom::NodeInfo
>&& aNodeInfo
)
40 : SVGGraphicsElementBase(std::move(aNodeInfo
)) {}
42 SVGElement
* SVGGraphicsElement::GetNearestViewportElement() {
43 return SVGContentUtils::GetNearestViewportElement(this);
46 SVGElement
* SVGGraphicsElement::GetFarthestViewportElement() {
47 return SVGContentUtils::GetOuterSVGElement(this);
50 static already_AddRefed
<SVGRect
> ZeroBBox(SVGGraphicsElement
& aOwner
) {
51 return MakeAndAddRef
<SVGRect
>(&aOwner
, gfx::Rect
{0, 0, 0, 0});
54 already_AddRefed
<SVGRect
> SVGGraphicsElement::GetBBox(
55 const SVGBoundingBoxOptions
& aOptions
) {
56 nsIFrame
* frame
= GetPrimaryFrame(FlushType::Layout
);
58 if (!frame
|| frame
->HasAnyStateBits(NS_FRAME_IS_NONDISPLAY
)) {
59 return ZeroBBox(*this);
61 ISVGDisplayableFrame
* svgframe
= do_QueryFrame(frame
);
64 if (!frame
->IsInSVGTextSubtree()) {
65 return ZeroBBox(*this);
68 // For <tspan>, <textPath>, the frame is an nsInlineFrame or
69 // nsBlockFrame, |svgframe| will be a nullptr.
70 // We implement their getBBox directly here instead of in
71 // SVGUtils::GetBBox, because SVGUtils::GetBBox is more
72 // or less used for other purpose elsewhere. e.g. gradient
73 // code assumes GetBBox of <tspan> returns the bbox of the
75 // TODO: cleanup this sort of usecase of SVGUtils::GetBBox,
76 // then move this code SVGUtils::GetBBox.
78 static_cast<SVGTextFrame
*>(nsLayoutUtils::GetClosestFrameOfType(
79 frame
->GetParent(), LayoutFrameType::SVGText
));
81 if (text
->HasAnyStateBits(NS_FRAME_IS_NONDISPLAY
)) {
82 return ZeroBBox(*this);
85 gfxRect rec
= text
->TransformFrameRectFromTextChild(
86 frame
->GetRectRelativeToSelf(), frame
);
88 // Should also add the |x|, |y| of the SVGTextFrame itself, since
89 // the result obtained by TransformFrameRectFromTextChild doesn't
91 rec
.x
+= float(text
->GetPosition().x
) / AppUnitsPerCSSPixel();
92 rec
.y
+= float(text
->GetPosition().y
) / AppUnitsPerCSSPixel();
94 return do_AddRef(new SVGRect(this, ToRect(rec
)));
97 if (!NS_SVGNewGetBBoxEnabled()) {
98 return do_AddRef(new SVGRect(
99 this, ToRect(SVGUtils::GetBBox(
100 frame
, SVGUtils::eBBoxIncludeFillGeometry
|
101 SVGUtils::eUseUserSpaceOfUseElement
))));
104 if (aOptions
.mFill
) {
105 flags
|= SVGUtils::eBBoxIncludeFillGeometry
;
107 if (aOptions
.mStroke
) {
108 flags
|= SVGUtils::eBBoxIncludeStroke
;
110 if (aOptions
.mMarkers
) {
111 flags
|= SVGUtils::eBBoxIncludeMarkers
;
113 if (aOptions
.mClipped
) {
114 flags
|= SVGUtils::eBBoxIncludeClipped
;
117 return do_AddRef(new SVGRect(this, {}));
119 if (flags
== SVGUtils::eBBoxIncludeMarkers
||
120 flags
== SVGUtils::eBBoxIncludeClipped
) {
121 flags
|= SVGUtils::eBBoxIncludeFillGeometry
;
123 flags
|= SVGUtils::eUseUserSpaceOfUseElement
;
124 return do_AddRef(new SVGRect(this, ToRect(SVGUtils::GetBBox(frame
, flags
))));
127 already_AddRefed
<SVGMatrix
> SVGGraphicsElement::GetCTM() {
128 if (auto* currentDoc
= GetComposedDoc()) {
129 // Flush all pending notifications so that our frames are up to date
130 currentDoc
->FlushPendingNotifications(FlushType::Layout
);
132 gfx::Matrix m
= SVGContentUtils::GetCTM(this);
133 RefPtr
<SVGMatrix
> mat
=
134 m
.IsSingular() ? nullptr : new SVGMatrix(ThebesMatrix(m
));
138 already_AddRefed
<SVGMatrix
> SVGGraphicsElement::GetScreenCTM() {
139 if (auto* currentDoc
= GetComposedDoc()) {
140 // Flush all pending notifications so that our frames are up to date
141 currentDoc
->FlushPendingNotifications(FlushType::Layout
);
143 gfx::Matrix m
= SVGContentUtils::GetScreenCTM(this);
144 RefPtr
<SVGMatrix
> mat
=
145 m
.IsSingular() ? nullptr : new SVGMatrix(ThebesMatrix(m
));
149 bool SVGGraphicsElement::IsSVGFocusable(bool* aIsFocusable
,
150 int32_t* aTabIndex
) {
151 // XXXedgar, maybe we could factor out the common code for SVG, HTML and
152 // MathML elements, see bug 1586011.
153 if (!IsInComposedDoc() || IsInDesignMode()) {
154 // In designMode documents we only allow focusing the document.
156 *aIsFocusable
= false;
160 *aTabIndex
= TabIndex();
161 // If a tabindex is specified at all, or the default tabindex is 0, we're
163 *aIsFocusable
= *aTabIndex
>= 0 || GetTabIndexAttrValue().isSome();
167 Focusable
SVGGraphicsElement::IsFocusableWithoutStyle(IsFocusableFlags
) {
169 IsSVGFocusable(&result
.mFocusable
, &result
.mTabIndex
);
173 } // namespace mozilla::dom