/* -*- 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/ShapeUtils.h" #include #include "nsCSSRendering.h" #include "nsMargin.h" #include "nsRuleNode.h" #include "nsStyleCoord.h" #include "nsStyleStruct.h" #include "SVGContentUtils.h" namespace mozilla { nscoord ShapeUtils::ComputeShapeRadius(const StyleShapeRadius aType, const nscoord aCenter, const nscoord aPosMin, const nscoord aPosMax) { nscoord dist1 = std::abs(aPosMin - aCenter); nscoord dist2 = std::abs(aPosMax - aCenter); nscoord length = 0; switch (aType) { case StyleShapeRadius::FarthestSide: length = dist1 > dist2 ? dist1 : dist2; break; case StyleShapeRadius::ClosestSide: length = dist1 > dist2 ? dist2 : dist1; break; } return length; } nsPoint ShapeUtils::ComputeCircleOrEllipseCenter(const StyleBasicShape* aBasicShape, const nsRect& aRefBox) { MOZ_ASSERT(aBasicShape->GetShapeType() == StyleBasicShapeType::Circle || aBasicShape->GetShapeType() == StyleBasicShapeType::Ellipse, "The basic shape must be circle() or ellipse!"); nsPoint topLeft, anchor; nsSize size(aRefBox.Size()); nsImageRenderer::ComputeObjectAnchorPoint(aBasicShape->GetPosition(), size, size, &topLeft, &anchor); return anchor + aRefBox.TopLeft(); } nscoord ShapeUtils::ComputeCircleRadius(const StyleBasicShape* aBasicShape, const nsPoint& aCenter, const nsRect& aRefBox) { MOZ_ASSERT(aBasicShape->GetShapeType() == StyleBasicShapeType::Circle, "The basic shape must be circle()!"); const nsTArray& coords = aBasicShape->Coordinates(); MOZ_ASSERT(coords.Length() == 1, "wrong number of arguments"); nscoord r = 0; if (coords[0].GetUnit() == eStyleUnit_Enumerated) { const auto styleShapeRadius = coords[0].GetEnumValue(); nscoord horizontal = ComputeShapeRadius(styleShapeRadius, aCenter.x, aRefBox.x, aRefBox.XMost()); nscoord vertical = ComputeShapeRadius(styleShapeRadius, aCenter.y, aRefBox.y, aRefBox.YMost()); r = styleShapeRadius == StyleShapeRadius::FarthestSide ? std::max(horizontal, vertical) : std::min(horizontal, vertical); } else { // We resolve percent value for circle() as defined here: // https://drafts.csswg.org/css-shapes/#funcdef-circle double referenceLength = SVGContentUtils::ComputeNormalizedHypotenuse(aRefBox.width, aRefBox.height); r = nsRuleNode::ComputeCoordPercentCalc(coords[0], NSToCoordRound(referenceLength)); } return r; } nsSize ShapeUtils::ComputeEllipseRadii(const StyleBasicShape* aBasicShape, const nsPoint& aCenter, const nsRect& aRefBox) { MOZ_ASSERT(aBasicShape->GetShapeType() == StyleBasicShapeType::Ellipse, "The basic shape must be ellipse()!"); const nsTArray& coords = aBasicShape->Coordinates(); MOZ_ASSERT(coords.Length() == 2, "wrong number of arguments"); nsSize radii; if (coords[0].GetUnit() == eStyleUnit_Enumerated) { const StyleShapeRadius radiusX = coords[0].GetEnumValue(); radii.width = ComputeShapeRadius(radiusX, aCenter.x, aRefBox.x, aRefBox.XMost()); } else { radii.width = nsRuleNode::ComputeCoordPercentCalc(coords[0], aRefBox.width); } if (coords[1].GetUnit() == eStyleUnit_Enumerated) { const StyleShapeRadius radiusY = coords[1].GetEnumValue(); radii.height = ComputeShapeRadius(radiusY, aCenter.y, aRefBox.y, aRefBox.YMost()); } else { radii.height = nsRuleNode::ComputeCoordPercentCalc(coords[1], aRefBox.height); } return radii; } /* static */ nsRect ShapeUtils::ComputeInsetRect(const StyleBasicShape* aBasicShape, const nsRect& aRefBox) { MOZ_ASSERT(aBasicShape->GetShapeType() == StyleBasicShapeType::Inset, "The basic shape must be inset()!"); const nsTArray& coords = aBasicShape->Coordinates(); MOZ_ASSERT(coords.Length() == 4, "wrong number of arguments"); nsMargin inset(nsRuleNode::ComputeCoordPercentCalc(coords[0], aRefBox.height), nsRuleNode::ComputeCoordPercentCalc(coords[1], aRefBox.width), nsRuleNode::ComputeCoordPercentCalc(coords[2], aRefBox.height), nsRuleNode::ComputeCoordPercentCalc(coords[3], aRefBox.width)); nsRect insetRect(aRefBox); insetRect.Deflate(inset); return insetRect; } /* static */ bool ShapeUtils::ComputeInsetRadii(const StyleBasicShape* aBasicShape, const nsRect& aInsetRect, const nsRect& aRefBox, nscoord aRadii[8]) { const nsStyleCorners& radius = aBasicShape->GetRadius(); return nsIFrame::ComputeBorderRadii(radius, aInsetRect.Size(), aRefBox.Size(), Sides(), aRadii); } /* static */ nsTArray ShapeUtils::ComputePolygonVertices(const StyleBasicShape* aBasicShape, const nsRect& aRefBox) { MOZ_ASSERT(aBasicShape->GetShapeType() == StyleBasicShapeType::Polygon, "The basic shape must be polygon()!"); const nsTArray& coords = aBasicShape->Coordinates(); MOZ_ASSERT(coords.Length() % 2 == 0 && coords.Length() >= 2, "Wrong number of arguments!"); nsTArray vertices(coords.Length() / 2); for (size_t i = 0; i + 1 < coords.Length(); i += 2) { vertices.AppendElement( nsPoint(nsRuleNode::ComputeCoordPercentCalc(coords[i], aRefBox.width), nsRuleNode::ComputeCoordPercentCalc(coords[i + 1], aRefBox.height)) + aRefBox.TopLeft()); } return vertices; } } // namespace mozilla