forked from mirrors/gecko-dev
		
	To address reviewer's comments in bug 1326409 comment 13 and comment 14. MozReview-Commit-ID: FiUXXk4O8N7
		
			
				
	
	
		
			173 lines
		
	
	
	
		
			6.4 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			173 lines
		
	
	
	
		
			6.4 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
/* -*- 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 <cstdlib>
 | 
						|
 | 
						|
#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<nsStyleCoord>& 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<StyleShapeRadius>();
 | 
						|
    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 <shape-radius> 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<nsStyleCoord>& 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<StyleShapeRadius>();
 | 
						|
    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<StyleShapeRadius>();
 | 
						|
    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<nsStyleCoord>& 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<nsPoint>
 | 
						|
ShapeUtils::ComputePolygonVertices(const StyleBasicShape* aBasicShape,
 | 
						|
                                   const nsRect& aRefBox)
 | 
						|
{
 | 
						|
  MOZ_ASSERT(aBasicShape->GetShapeType() == StyleBasicShapeType::Polygon,
 | 
						|
             "The basic shape must be polygon()!");
 | 
						|
 | 
						|
  const nsTArray<nsStyleCoord>& coords = aBasicShape->Coordinates();
 | 
						|
  MOZ_ASSERT(coords.Length() % 2 == 0 &&
 | 
						|
             coords.Length() >= 2, "Wrong number of arguments!");
 | 
						|
 | 
						|
  nsTArray<nsPoint> 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
 |