package shape import ( "math" "oss.terrastruct.com/d2/lib/geo" "oss.terrastruct.com/util-go/go2" ) type shapeCircle struct { *baseShape } func NewCircle(box *geo.Box) Shape { shape := shapeCircle{ baseShape: &baseShape{ Type: CIRCLE_TYPE, Box: box, }, } shape.FullShape = go2.Pointer(Shape(shape)) return shape } func (s shapeCircle) GetInnerBox() *geo.Box { width := s.Box.Width height := s.Box.Height insideTL := s.GetInsidePlacement(width, height, 0, 0) tl := s.Box.TopLeft.Copy() width -= 2 * (insideTL.X - tl.X) height -= 2 * (insideTL.Y - tl.Y) return geo.NewBox(&insideTL, width, height) } func (s shapeCircle) AspectRatio1() bool { return true } func (s shapeCircle) GetDimensionsToFit(width, height, paddingX, paddingY float64) (float64, float64) { length := math.Max(width+paddingX, height+paddingY) diameter := math.Ceil(math.Sqrt2 * length) return diameter, diameter } func (s shapeCircle) GetInsidePlacement(width, height, paddingX, paddingY float64) geo.Point { project45 := 1 / math.Sqrt2 r := s.Box.Width / 2 // we want to offset r-padding/2 away from the center return geo.Point{ X: s.Box.TopLeft.X + math.Ceil(r-project45*(r-paddingX/2)), Y: s.Box.TopLeft.Y + math.Ceil(r-project45*(r-paddingY/2)), } } func (s shapeCircle) Perimeter() []geo.Intersectable { return []geo.Intersectable{geo.NewEllipse(s.Box.Center(), s.Box.Width/2, s.Box.Height/2)} } func (s shapeCircle) GetDefaultPadding() (paddingX, paddingY float64) { return defaultPadding / math.Sqrt2, defaultPadding / math.Sqrt2 }