a
    ZXh                     @   sl  d Z ddlmZmZmZ ddlmZ ddlZddlm	Z	 zddl
Z
W n" eefyf   ddlm
Z
 Y n0 e
jZdZe	dg d	Zg d
ZdddZdd Ze
e
je
je
je
je
je
jde
je
je
je
jddd Ze
e
je
je
je
je
je
jde
je
je
jddddZdZdZe
je
je
e
je
je
je
jddd Ze
je
je
e
je
je
jddd Z d d! Z!e
e
je
je
je
je
je
je
je
je
jd"e
je
je
je
je
je
je
je
jd#d$d% Z"d&d' Z#e
e
je
je
je
je
jd(e
je
je
je
jd)d*d+ Z$d,d- Z%d.d/ Z&e
e
je
je
je
je
je
jde
je
je
je
je
je
jd0d1d2 Z'd3d4 Z(d5d6 Z)d7d8 Z*d9d: Z+d;d< Z,d=d> Z-e
je
je
je
je
je
je
je
je
jd?d@dA Z.e
e
je
je
je
je
je
je
je
je
je
jdBe
je
je
je
je
jdCdDdE Z/dFdG Z0dHdI Z1e
je
je
je
je
je
je
je
je
je
je
je
je
je
jdJdKdL Z2ddMlm3Z3m4Z4m5Z5m6Z6 e3fdNdOZ7dPdQ Z8dRdS Z9dTdU Z:e
je
je
je
je
je
je
je
je
je
jdVdWdX Z;dYdZ Z<d[d\ Z=e
je
je
je
je
je
je
je
je
je
jd]d^d_ Z>d`da Z?dbdc Z@ddde ZAe
e
je
je
je
je
je
je
jdfe
je
je
je
jdgdhdi ZBdjdk ZCdldm ZDdndo ZEdpdq ZFdrds ZGdtdu ZHdvdw ZIdxdy ZJdzd{ ZKdd}d~ZLdd ZMdd ZNdd ZOdd ZPdd ZQeRdkrhddlSZSddlTZTeSUeTV jW dS )zNfontTools.misc.bezierTools.py -- tools for working with Bezier path segments.
    )
calcBoundssectRectrectArea)IdentityN)
namedtuple)cythong&.>Intersectionptt1t2)approximateCubicArcLengthapproximateCubicArcLengthCapproximateQuadraticArcLengthapproximateQuadraticArcLengthCcalcCubicArcLengthcalcCubicArcLengthCcalcQuadraticArcLengthcalcQuadraticArcLengthCcalcCubicBoundscalcQuadraticBounds	splitLinesplitQuadratic
splitCubicsplitQuadraticAtTsplitCubicAtTsplitCubicAtTCsplitCubicIntoTwoAtTCsolveQuadratic
solveCubicquadraticPointAtTcubicPointAtTcubicPointAtTClinePointAtTsegmentPointAtTlineLineIntersectionscurveLineIntersectionscurveCurveIntersectionssegmentSegmentIntersections{Gzt?c                 C   s    t t|  t| t| t| |S )a  Calculates the arc length for a cubic Bezier segment.

    Whereas :func:`approximateCubicArcLength` approximates the length, this
    function calculates it by "measuring", recursively dividing the curve
    until the divided segments are shorter than ``tolerance``.

    Args:
        pt1,pt2,pt3,pt4: Control points of the Bezier as 2D tuples.
        tolerance: Controls the precision of the calcuation.

    Returns:
        Arc length value.
    )r   complex)pt1pt2pt3pt4	tolerance r0   _/var/www/viveiro_mudafortebrasil/venv/lib/python3.9/site-packages/fontTools/misc/bezierTools.pyr   8   s    r   c                 C   s\   | d||   | d }|| | |  d }| | | d || |f||| || d |ffS )N   g      ?      ?r0   )p0p1p2p3ZmidZderiv3r0   r0   r1   _split_cubic_into_twoK   s
    r8   )r4   r5   r6   r7   )multarchboxc           	      C   s   t || }t || t ||  t ||  }||  t |krL|| d S t||||\}}t| g|R  t| g|R   S d S Nr3   )absEPSILONr8   _calcCubicArcLengthCRecurse)	r9   r4   r5   r6   r7   r:   r;   ZoneZtwor0   r0   r1   r?   T   s    	$r?   r+   r,   r-   r.   )r/   r9   c                 C   s   dd|  }t || |||S )zCalculates the arc length for a cubic Bezier segment.

    Args:
        pt1,pt2,pt3,pt4: Control points of the Bezier as complex numbers.
        tolerance: Controls the precision of the calcuation.

    Returns:
        Arc length value.
          ?g      ?)r?   )r+   r,   r-   r.   r/   r9   r0   r0   r1   r   h   s    r      g|=v1v2c                 C   s   | |   jS N)	conjugaterealrC   r0   r0   r1   _dot   s    rI   xc                 C   s(   | t | d d  d t | d  S )N      )mathsqrtasinhrJ   r0   r0   r1   _intSecAtan   s    rQ   c                 C   s   t t|  t| t| S )a  Calculates the arc length for a quadratic Bezier segment.

    Args:
        pt1: Start point of the Bezier as 2D tuple.
        pt2: Handle point of the Bezier as 2D tuple.
        pt3: End point of the Bezier as 2D tuple.

    Returns:
        Arc length value.

    Example::

        >>> calcQuadraticArcLength((0, 0), (0, 0), (0, 0)) # empty segment
        0.0
        >>> calcQuadraticArcLength((0, 0), (50, 0), (80, 0)) # collinear points
        80.0
        >>> calcQuadraticArcLength((0, 0), (0, 50), (0, 80)) # collinear points vertical
        80.0
        >>> calcQuadraticArcLength((0, 0), (50, 20), (100, 40)) # collinear points
        107.70329614269008
        >>> calcQuadraticArcLength((0, 0), (0, 100), (100, 0))
        154.02976155645263
        >>> calcQuadraticArcLength((0, 0), (0, 50), (100, 0))
        120.21581243984076
        >>> calcQuadraticArcLength((0, 0), (50, -10), (80, 50))
        102.53273816445825
        >>> calcQuadraticArcLength((0, 0), (40, 0), (-40, 0)) # collinear points, control point outside
        66.66666666666667
        >>> calcQuadraticArcLength((0, 0), (40, 0), (0, 0)) # collinear points, looping back
        40.0
    )r   r*   r+   r,   r-   r0   r0   r1   r      s     r   )r+   r,   r-   d0d1dn)scaleorigDistabx0x1Lenc                 C   s   ||  }|| }|| }|d }t |}|dkr<t ||  S t||}t |tk rt||dkrlt ||  S t |t | }	}
|	|	 |
|
  |	|
  S t||| }t||| }t dt|t|  | |||   }|S )a$  Calculates the arc length for a quadratic Bezier segment.

    Args:
        pt1: Start point of the Bezier as a complex number.
        pt2: Handle point of the Bezier as a complex number.
        pt3: End point of the Bezier as a complex number.

    Returns:
        Arc length value.
    y              ?        r   rL   )r=   rI   epsilonrQ   )r+   r,   r-   rS   rT   rU   rV   rW   rX   rY   rZ   r[   r\   r]   r0   r0   r1   r      s"     
(r   c                 C   s   t t|  t| t| S )a  Calculates the arc length for a quadratic Bezier segment.

    Uses Gauss-Legendre quadrature for a branch-free approximation.
    See :func:`calcQuadraticArcLength` for a slower but more accurate result.

    Args:
        pt1: Start point of the Bezier as 2D tuple.
        pt2: Handle point of the Bezier as 2D tuple.
        pt3: End point of the Bezier as 2D tuple.

    Returns:
        Approximate arc length value.
    )r   r*   rR   r0   r0   r1   r      s    r   rR   )v0rD   rE   c                 C   sT   t d|  d|  d|  }t ||  d }t d|  d|  d|  }|| | S )a  Calculates the arc length for a quadratic Bezier segment.

    Uses Gauss-Legendre quadrature for a branch-free approximation.
    See :func:`calcQuadraticArcLength` for a slower but more accurate result.

    Args:
        pt1: Start point of the Bezier as a complex number.
        pt2: Handle point of the Bezier as a complex number.
        pt3: End point of the Bezier as a complex number.

    Returns:
        Approximate arc length value.
    g̔xb߿gb?gFVW?gqq?gFVWg̔xb?r=   )r+   r,   r-   r`   rD   rE   r0   r0   r1   r      s    !r   c                    s   t | ||\\ \\ d }d }g }|dkrJ| |  |dkrb| |   fdd|D | |g }t|S )a  Calculates the bounding rectangle for a quadratic Bezier segment.

    Args:
        pt1: Start point of the Bezier as a 2D tuple.
        pt2: Handle point of the Bezier as a 2D tuple.
        pt3: End point of the Bezier as a 2D tuple.

    Returns:
        A four-item tuple representing the bounding rectangle ``(xMin, yMin, xMax, yMax)``.

    Example::

        >>> calcQuadraticBounds((0, 0), (50, 100), (100, 0))
        (0, 0, 100, 50.0)
        >>> calcQuadraticBounds((0, 0), (100, 0), (100, 100))
        (0.0, 0.0, 100, 100)
           @r   c                    sT   g | ]L}d |  krdk rn q | | |   | | |   fqS r   rM   r0   .0taxaybxbycxcyr0   r1   
<listcomp>D  s   z'calcQuadraticBounds.<locals>.<listcomp>)calcQuadraticParametersappendr   )r+   r,   r-   Zax2Zay2rootspointsr0   rg   r1   r   *  s    r   c                 C   s   t t|  t| t| t| S )a  Approximates the arc length for a cubic Bezier segment.

    Uses Gauss-Lobatto quadrature with n=5 points to approximate arc length.
    See :func:`calcCubicArcLength` for a slower but more accurate result.

    Args:
        pt1,pt2,pt3,pt4: Control points of the Bezier as 2D tuples.

    Returns:
        Arc length value.

    Example::

        >>> approximateCubicArcLength((0, 0), (25, 100), (75, 100), (100, 0))
        190.04332968932817
        >>> approximateCubicArcLength((0, 0), (50, 0), (100, 50), (100, 100))
        154.8852074945903
        >>> approximateCubicArcLength((0, 0), (50, 0), (100, 0), (150, 0)) # line; exact result should be 150.
        149.99999999999991
        >>> approximateCubicArcLength((0, 0), (50, 0), (100, 0), (-50, 0)) # cusp; exact result should be 150.
        136.9267662156362
        >>> approximateCubicArcLength((0, 0), (50, 0), (100, -50), (-50, 0)) # cusp
        154.80848416537057
    )r   r*   r@   r0   r0   r1   r   L  s    r   )r`   rD   rE   v3v4c           	      C   s   t ||  d }t d|  d|  d|  d|  }t ||  | | d }t d|  d|  d|  d|  }t || d }|| | | | S )	zApproximates the arc length for a cubic Bezier segment.

    Args:
        pt1,pt2,pt3,pt4: Control points of the Bezier as complex numbers.

    Returns:
        Arc length value.
    g333333?gc1g85$t?gu|Y?g#$?g?g#$gc1?ra   )	r+   r,   r-   r.   r`   rD   rE   rs   rt   r0   r0   r1   r   j  s,    r   c                    s   t | |||\\ \\\ d }d }d }d }dd t||D }dd t||D }	||	 }
 fdd|
D | |g }t|S )aX  Calculates the bounding rectangle for a quadratic Bezier segment.

    Args:
        pt1,pt2,pt3,pt4: Control points of the Bezier as 2D tuples.

    Returns:
        A four-item tuple representing the bounding rectangle ``(xMin, yMin, xMax, yMax)``.

    Example::

        >>> calcCubicBounds((0, 0), (25, 100), (75, 100), (100, 0))
        (0, 0, 100, 75.0)
        >>> calcCubicBounds((0, 0), (50, 0), (100, 50), (100, 100))
        (0.0, 0.0, 100, 100)
        >>> print("%f %f %f %f" % calcCubicBounds((50, 0), (0, 100), (100, 100), (50, 0)))
        35.566243 0.000000 64.433757 75.000000
          @rb   c                 S   s(   g | ] }d |  krdk rn q|qS rc   r0   rd   r0   r0   r1   rn         z#calcCubicBounds.<locals>.<listcomp>c                 S   s(   g | ] }d |  krdk rn q|qS rc   r0   rd   r0   r0   r1   rn     rv   c                    s\   g | ]T} | | | | |  |   | | | | |  |   fqS r0   r0   rd   rh   ri   rj   rk   rl   rm   dxdyr0   r1   rn     s   &&)calcCubicParametersr   r   )r+   r,   r-   r.   Zax3Zay3Zbx2Zby2ZxRootsZyRootsrq   rr   r0   rw   r1   r     s    &r   c                 C   s   | \}}|\}}|| }|| }	|}
|}||	f| }|dkrF| |fgS ||
|f|  | }d|  krndk rn n(|| |
 |	| | f}| |f||fgS | |fgS dS )a  Split a line at a given coordinate.

    Args:
        pt1: Start point of line as 2D tuple.
        pt2: End point of line as 2D tuple.
        where: Position at which to split the line.
        isHorizontal: Direction of the ray splitting the line. If true,
            ``where`` is interpreted as a Y coordinate; if false, then
            ``where`` is interpreted as an X coordinate.

    Returns:
        A list of two line segments (each line segment being two 2D tuples)
        if the line was successfully split, or a list containing the original
        line.

    Example::

        >>> printSegments(splitLine((0, 0), (100, 100), 50, True))
        ((0, 0), (50, 50))
        ((50, 50), (100, 100))
        >>> printSegments(splitLine((0, 0), (100, 100), 100, True))
        ((0, 0), (100, 100))
        >>> printSegments(splitLine((0, 0), (100, 100), 0, True))
        ((0, 0), (0, 0))
        ((0, 0), (100, 100))
        >>> printSegments(splitLine((0, 0), (100, 100), 0, False))
        ((0, 0), (0, 0))
        ((0, 0), (100, 100))
        >>> printSegments(splitLine((100, 0), (0, 0), 50, False))
        ((100, 0), (50, 0))
        ((50, 0), (0, 0))
        >>> printSegments(splitLine((0, 100), (0, 0), 50, True))
        ((0, 100), (0, 50))
        ((0, 50), (0, 0))
    r   rM   Nr0   )r+   r,   whereisHorizontalZpt1xZpt1yZpt2xZpt2yrh   ri   rj   rk   rY   rf   ZmidPtr0   r0   r1   r     s    $
r   c           	      C   sd   t | ||\}}}t|| || || | }tdd |D }|sP| ||fgS t|||g|R  S )a  Split a quadratic Bezier curve at a given coordinate.

    Args:
        pt1,pt2,pt3: Control points of the Bezier as 2D tuples.
        where: Position at which to split the curve.
        isHorizontal: Direction of the ray splitting the curve. If true,
            ``where`` is interpreted as a Y coordinate; if false, then
            ``where`` is interpreted as an X coordinate.

    Returns:
        A list of two curve segments (each curve segment being three 2D tuples)
        if the curve was successfully split, or a list containing the original
        curve.

    Example::

        >>> printSegments(splitQuadratic((0, 0), (50, 100), (100, 0), 150, False))
        ((0, 0), (50, 100), (100, 0))
        >>> printSegments(splitQuadratic((0, 0), (50, 100), (100, 0), 50, False))
        ((0, 0), (25, 50), (50, 50))
        ((50, 50), (75, 50), (100, 0))
        >>> printSegments(splitQuadratic((0, 0), (50, 100), (100, 0), 25, False))
        ((0, 0), (12.5, 25), (25, 37.5))
        ((25, 37.5), (62.5, 75), (100, 0))
        >>> printSegments(splitQuadratic((0, 0), (50, 100), (100, 0), 25, True))
        ((0, 0), (7.32233, 14.6447), (14.6447, 25))
        ((14.6447, 25), (50, 75), (85.3553, 25))
        ((85.3553, 25), (92.6777, 14.6447), (100, -7.10543e-15))
        >>> # XXX I'm not at all sure if the following behavior is desirable:
        >>> printSegments(splitQuadratic((0, 0), (50, 100), (100, 0), 50, True))
        ((0, 0), (25, 50), (50, 50))
        ((50, 50), (50, 50), (50, 50))
        ((50, 50), (75, 50), (100, 0))
    c                 s   s*   | ]"}d |  krdk rn q|V  qdS r   rM   Nr0   rd   r0   r0   r1   	<genexpr>"  rv   z!splitQuadratic.<locals>.<genexpr>)ro   r   sorted_splitQuadraticAtT)	r+   r,   r-   r{   r|   rY   rZ   c	solutionsr0   r0   r1   r     s    #r   c                 C   sr   t | |||\}}}}	t|| || || |	| | }
tdd |
D }
|
s\| |||fgS t||||	g|
R  S )a  Split a cubic Bezier curve at a given coordinate.

    Args:
        pt1,pt2,pt3,pt4: Control points of the Bezier as 2D tuples.
        where: Position at which to split the curve.
        isHorizontal: Direction of the ray splitting the curve. If true,
            ``where`` is interpreted as a Y coordinate; if false, then
            ``where`` is interpreted as an X coordinate.

    Returns:
        A list of two curve segments (each curve segment being four 2D tuples)
        if the curve was successfully split, or a list containing the original
        curve.

    Example::

        >>> printSegments(splitCubic((0, 0), (25, 100), (75, 100), (100, 0), 150, False))
        ((0, 0), (25, 100), (75, 100), (100, 0))
        >>> printSegments(splitCubic((0, 0), (25, 100), (75, 100), (100, 0), 50, False))
        ((0, 0), (12.5, 50), (31.25, 75), (50, 75))
        ((50, 75), (68.75, 75), (87.5, 50), (100, 0))
        >>> printSegments(splitCubic((0, 0), (25, 100), (75, 100), (100, 0), 25, True))
        ((0, 0), (2.29379, 9.17517), (4.79804, 17.5085), (7.47414, 25))
        ((7.47414, 25), (31.2886, 91.6667), (68.7114, 91.6667), (92.5259, 25))
        ((92.5259, 25), (95.202, 17.5085), (97.7062, 9.17517), (100, 1.77636e-15))
    c                 s   s*   | ]"}d |  krdk rn q|V  qdS r}   r0   rd   r0   r0   r1   r~   G  rv   zsplitCubic.<locals>.<genexpr>)rz   r   r   _splitCubicAtT)r+   r,   r-   r.   r{   r|   rY   rZ   r   rU   r   r0   r0   r1   r   (  s    r   c                 G   s&   t | ||\}}}t|||g|R  S )a  Split a quadratic Bezier curve at one or more values of t.

    Args:
        pt1,pt2,pt3: Control points of the Bezier as 2D tuples.
        *ts: Positions at which to split the curve.

    Returns:
        A list of curve segments (each curve segment being three 2D tuples).

    Examples::

        >>> printSegments(splitQuadraticAtT((0, 0), (50, 100), (100, 0), 0.5))
        ((0, 0), (25, 50), (50, 50))
        ((50, 50), (75, 50), (100, 0))
        >>> printSegments(splitQuadraticAtT((0, 0), (50, 100), (100, 0), 0.5, 0.75))
        ((0, 0), (25, 50), (50, 50))
        ((50, 50), (62.5, 50), (75, 37.5))
        ((75, 37.5), (87.5, 25), (100, 0))
    )ro   r   )r+   r,   r-   tsrY   rZ   r   r0   r0   r1   r   M  s    r   c           
      G   sj   t | |||\}}}}t||||g|R  }	| g|	d dd R |	d< g |	d dd |R |	d< |	S )a   Split a cubic Bezier curve at one or more values of t.

    Args:
        pt1,pt2,pt3,pt4: Control points of the Bezier as 2D tuples.
        *ts: Positions at which to split the curve.

    Returns:
        A list of curve segments (each curve segment being four 2D tuples).

    Examples::

        >>> printSegments(splitCubicAtT((0, 0), (25, 100), (75, 100), (100, 0), 0.5))
        ((0, 0), (12.5, 50), (31.25, 75), (50, 75))
        ((50, 75), (68.75, 75), (87.5, 50), (100, 0))
        >>> printSegments(splitCubicAtT((0, 0), (25, 100), (75, 100), (100, 0), 0.5, 0.75))
        ((0, 0), (12.5, 50), (31.25, 75), (50, 75))
        ((50, 75), (59.375, 75), (68.75, 68.75), (77.3438, 56.25))
        ((77.3438, 56.25), (85.9375, 43.75), (93.75, 25), (100, 0))
    r   rM   N)rz   r   )
r+   r,   r-   r.   r   rY   rZ   r   rU   splitr0   r0   r1   r   e  s
    r   )r+   r,   r-   r.   rY   rZ   r   rU   c           	      g   s6   t | |||\}}}}t||||g|R  E dH  dS )a  Split a cubic Bezier curve at one or more values of t.

    Args:
        pt1,pt2,pt3,pt4: Control points of the Bezier as complex numbers..
        *ts: Positions at which to split the curve.

    Yields:
        Curve segments (each curve segment being four complex numbers).
    N)calcCubicParametersC_splitCubicAtTC)	r+   r,   r-   r.   r   rY   rZ   r   rU   r0   r0   r1   r     s    r   )rf   r+   r,   r-   r.   pointAtToff1off2)r   _1_t_1_t_2_2_t_1_tc                 C   s   || }d| }|| }d| | }|| |  d|| | || |    || |  }	||  ||  ||  }
|| ||  ||  }| ||  |  }||| |  }| ||
|	f|	|||ffS )a  Split a cubic Bezier curve at t.

    Args:
        pt1,pt2,pt3,pt4: Control points of the Bezier as complex numbers.
        t: Position at which to split the curve.

    Returns:
        A tuple of two curve segments (each curve segment being four complex numbers).
    rM   rL   r2   r0   )r+   r,   r-   r.   rf   r   r   r   r   r   r   r   r0   r0   r1   r     s    2r   c                 G   s  t |}g }|dd |d | \}}|\}}|\}	}
tt|d D ]}|| }||d  }|| }|| }|| }|| }d| | | | }d| | | | }|| }|| ||  |	 }|| ||  |
 }t||f||f||f\}}}||||f qJ|S )Nr   r^   rA   rM   rL   )listinsertrp   rangelencalcQuadraticPoints)rY   rZ   r   r   segmentsrh   ri   rj   rk   rl   rm   ir   r   deltadelta_2a1xa1yb1xb1yt1_2c1xc1yr+   r,   r-   r0   r0   r1   r     s,    
r   c           "      G   s  t |}|dd |d g }| \}}|\}}	|\}
}|\}}tt|d D ](}|| }||d  }|| }|| }|| }|| }|| }|| }|| }d| | | | }d| | |	 | }d| | |
 d| |  | }d|	 | | d| |  | }|| ||  |
|  | }|| |	|  ||  | }t||f||f||f||f\}}} }!|||| |!f qR|S Nr   r^   rA   rM   r2   rL   )r   r   rp   r   r   calcCubicPoints)"rY   rZ   r   rU   r   r   rh   ri   rj   rk   rl   rm   rx   ry   r   r   r   r   r   delta_3r   t1_3r   r   r   r   r   r   Zd1xZd1yr+   r,   r-   r.   r0   r0   r1   r     s:    
  r   )rY   rZ   r   rU   r   r   r   r   r   a1b1c1rT   c                 g   s   t |}|dd |d tt|d D ]}|| }||d  }|| }|| }	||	 }
|| }|| }| |
 }d|  | | |	 }d| | | d|  |  | }| | ||  ||  | }t||||\}}}}||||fV  q.d S r   )r   r   rp   r   r   calcCubicPointsC)rY   rZ   r   rU   r   r   r   r   r   r   r   r   r   r   r   r   rT   r+   r,   r-   r.   r0   r0   r1   r     s"    
 r   )rO   acoscospic                 C   s~   t | tk r,t |tk rg }qz| | g}nN|| d|  |  }|dkrv||}| | d |  | | d |  g}ng }|S )uK  Solve a quadratic equation.

    Solves *a*x*x + b*x + c = 0* where a, b and c are real.

    Args:
        a: coefficient of *x²*
        b: coefficient of *x*
        c: constant term

    Returns:
        A list of roots. Note that the returned list is neither guaranteed to
        be sorted nor to contain unique values!
          @r^   rb   r=   r_   )rY   rZ   r   rO   rq   ZDDZrDDr0   r0   r1   r   /  s    &r   c                 C   s  t | tk rt|||S t| } ||  }||  }||  }|| d|  d }d| | | d| |  d|  d }|| }	|| | }
|	tk rdn|	}	t |
tk rdn|
}
|	|
 }|	dkr|
dkrt| d t}|||gS |td kr@ttt|t	|
 d	d
}dt	| }|d }|t
|d  | }|t
|dt  d  | }|t
|dt  d  | }t|||g\}}}|| tk r|| tk rt|| | d t } }}n~|| tk rt|| d t }}t|t}nN|| tk rt|t}t|| d t }}nt|t}t|t}t|t}|||gS tt	|t | d}|||  }|dkrr| }t||d  t}|gS dS )ut  Solve a cubic equation.

    Solves *a*x*x*x + b*x*x + c*x + d = 0* where a, b, c and d are real.

    Args:
        a: coefficient of *x³*
        b: coefficient of *x²*
        c: coefficient of *x*
        d: constant term

    Returns:
        A list of roots. Note that the returned list is neither guaranteed to
        be sorted nor to contain unique values!

    Examples::

        >>> solveCubic(1, 1, -6, 0)
        [-3.0, -0.0, 2.0]
        >>> solveCubic(-10.0, -9.0, 48.0, -29.0)
        [-2.9, 1.0, 1.0]
        >>> solveCubic(-9.875, -9.0, 47.625, -28.75)
        [-2.911392, 1.0, 1.0]
        >>> solveCubic(1.0, -4.5, 6.75, -3.375)
        [1.5, 1.5, 1.5]
        >>> solveCubic(-12.0, 18.0, -9.0, 1.50023651123)
        [0.5, 0.5, 0.5]
        >>> solveCubic(
        ...     9.0, 0.0, 0.0, -7.62939453125e-05
        ... ) == [-0.0, -0.0, -0.0]
        True
    ru   g      "@rb   g      ;@g      K@r   r^   r3   rA   g      g       r   UUUUUU?N)r=   r_   r   floatroundepsilonDigitsr   maxminrO   r   r   r   pow)rY   rZ   r   rU   r   Za2a3QRZR2ZQ3ZR2_Q3rK   thetaZrQ2Za1_3r[   r\   x2r0   r0   r1   r   P  sT    &(
 





r   c                 C   s^   |\}}|\}}| \}}|| d }	|| d }
|| |	 }|| |
 }||f|	|
f||ffS )Nrb   r0   )r+   r,   r-   r   y2x3y3rl   rm   rj   rk   rh   ri   r0   r0   r1   ro     s    ro   c                 C   s   |\}}|\}}|\}}	| \}
}||
 d }|| d }|| d | }|| d | }||
 | | }|	| | | }||f||f||f|
|ffS Nru   r0   )r+   r,   r-   r.   r   r   r   r   x4y4rx   ry   rl   rm   rj   rk   rh   ri   r0   r0   r1   rz     s    rz   )r+   r,   r-   r.   rY   rZ   r   c                 C   s8   ||  d }|| d | }||  | | }|||| fS r   r0   )r+   r,   r-   r.   r   rZ   rY   r0   r0   r1   r     s    r   c                 C   sf   | \}}|\}}|\}}|}	|}
|d | }|d | }|| | }|| | }|	|
f||f||ffS r<   r0   )rY   rZ   r   rh   ri   rj   rk   rl   rm   r\   y1r   r   r   r   r0   r0   r1   r     s    r   c                 C   s   | \}}|\}}|\}}	|\}
}|
}|}|d |
 }|	d | }|| d | }||	 d | }||
 | | }|| |	 | }||f||f||f||ffS r   r0   )rY   rZ   r   rU   rh   ri   rj   rk   rl   rm   rx   ry   r\   r   r   r   r   r   r   r   r0   r0   r1   r     s    r   rY   rZ   r   rU   r6   r7   Zp4c                 C   s8   |d | }|| d | }| | | | }||||fS )Nr   r0   r   r0   r0   r1   r     s    r   c                 C   s8   | d d|  |d |  | d d|  |d |  fS )zFinds the point at time `t` on a line.

    Args:
        pt1, pt2: Coordinates of the line as 2D tuples.
        t: The time along the line.

    Returns:
        A 2D tuple with the coordinates of the point.
    r   rM   r0   )r+   r,   rf   r0   r0   r1   r#     s    
r#   c                 C   s   d| d|  | d  dd|  | |d   || |d   }d| d|  | d  dd|  | |d   || |d   }||fS )zFinds the point at time `t` on a quadratic curve.

    Args:
        pt1, pt2, pt3: Coordinates of the curve as 2D tuples.
        t: The time along the curve.

    Returns:
        A 2D tuple with the coordinates of the point.
    rM   r   rL   r0   )r+   r,   r-   rf   rK   yr0   r0   r1   r      s    
@@r    c           
      C   s   || }d| }|| }|| | d  d|| |d  || |d     || |d   }|| | d  d|| |d  || |d     || |d   }	||	fS )zFinds the point at time `t` on a cubic curve.

    Args:
        pt1, pt2, pt3, pt4: Coordinates of the curve as 2D tuples.
        t: The time along the curve.

    Returns:
        A 2D tuple with the coordinates of the point.
    rM   r   r2   r0   )
r+   r,   r-   r.   rf   r   r   r   rK   r   r0   r0   r1   r!   ,  s     
""r!   )rf   r+   r,   r-   r.   )r   r   r   c                 C   sL   || }d| }|| }|| |  d|| | || |    || |  S )zFinds the point at time `t` on a cubic curve.

    Args:
        pt1, pt2, pt3, pt4: Coordinates of the curve as complex numbers.
        t: The time along the curve.

    Returns:
        A complex number with the coordinates of the point.
    rM   r2   r0   )r+   r,   r-   r.   rf   r   r   r   r0   r0   r1   r"   F  s    r"   c                 C   sf   t | dkrtg | |R  S t | dkr<tg | |R  S t | dkrZtg | |R  S tdd S NrL   r2      Unknown curve degree)r   r#   r    r!   
ValueError)segrf   r0   r0   r1   r$   _  s    r$   c           	      C   sx   | \}}|\}}|\}}t || tk r<t || tk r<dS t || t || krd|| ||  S || ||  S d S )Nr   r   )	ser
   ZsxZsyexZeyZpxpyr0   r0   r1   _line_t_of_ptn  s     r   c                 C   sR   | d |d  |d |d   }| d |d  |d |d   }|dkoN|dk S )Nr   rM   r^   r0   )rY   rZ   originZxDiffZyDiffr0   r0   r1   '_both_points_are_on_same_side_of_origin|  s      r   c                 C   s  | \}}|\}}|\}}	|\}
}t ||
rHt ||rHt ||sHg S t |	|rpt ||rpt ||	spg S t ||
rt |	|rg S t ||rt ||rg S t ||r|}||	 |
|  }|||  |	 }||f}t|t| ||t|||dgS t ||
r\|}|| ||  }|||  | }||f}t|t| ||t|||dgS || ||  }||	 |
|  }t ||rg S || | ||  |	 ||  }|||  | }||f}t||| rt|||rt|t| ||t|||dgS g S )a  Finds intersections between two line segments.

    Args:
        s1, e1: Coordinates of the first line as 2D tuples.
        s2, e2: Coordinates of the second line as 2D tuples.

    Returns:
        A list of ``Intersection`` objects, each object having ``pt``, ``t1``
        and ``t2`` attributes containing the intersection point, time on first
        segment and time on second segment respectively.

    Examples::

        >>> a = lineLineIntersections( (310,389), (453, 222), (289, 251), (447, 367))
        >>> len(a)
        1
        >>> intersection = a[0]
        >>> intersection.pt
        (374.44882952482897, 313.73458370177315)
        >>> (intersection.t1, intersection.t2)
        (0.45069111555824465, 0.5408153767394238)
    r	   )rN   iscloser   r   r   )s1e1s2e2Zs1xZs1yZe1xZe1yZs2xZs2yZe2xZe2yrK   Zslope34r   r
   Zslope12r0   r0   r1   r%     sr    





 
r%   c                 C   sT   | d }| d }t |d |d  |d |d  }t| |d  |d  S )Nr   r   rM   )rN   atan2r   rotate	translate)segmentstartendZangler0   r0   r1   _alignment_transformation  s    $r   c                 C   s   t || }t| dkrBt| \}}}t|d |d |d }nDt| dkr~t| \}}}}t|d |d |d |d }ntdtdd |D S )Nr2   rM   r   r   c                 s   s*   | ]"}d |  krdkrn q|V  qdS )r^   rM   Nr0   re   r   r0   r0   r1   r~     rv   z._curve_line_intersections_t.<locals>.<genexpr>)	r   transformPointsr   ro   r   rz   r   r   r   )curvelineZaligned_curverY   rZ   r   intersectionsrU   r0   r0   r1   _curve_line_intersections_t  s     r   c                 C   s   t | dkrt}nt | dkr$t}ntdg }t| |D ]N}|g | |R  }tg ||R  }tg ||R  }|t|||d q:|S )a  Finds intersections between a curve and a line.

    Args:
        curve: List of coordinates of the curve segment as 2D tuples.
        line: List of coordinates of the line segment as 2D tuples.

    Returns:
        A list of ``Intersection`` objects, each object having ``pt``, ``t1``
        and ``t2`` attributes containing the intersection point, time on first
        segment and time on second segment respectively.

    Examples::
        >>> curve = [ (100, 240), (30, 60), (210, 230), (160, 30) ]
        >>> line  = [ (25, 260), (230, 20) ]
        >>> intersections = curveLineIntersections(curve, line)
        >>> len(intersections)
        3
        >>> intersections[0].pt
        (84.9000930760723, 189.87306176459828)
    r2   r   r   r	   )	r   r    r!   r   r   r   r#   rp   r   )r   r   ZpointFinderr   rf   r
   Zline_tr0   r0   r1   r&     s    r&   c                 C   s4   t | dkrt|  S t | dkr(t|  S tdd S )Nr2   r   r   )r   r   r   r   )r   r0   r0   r1   _curve_bounds  s
    r   c                 C   sx   t | dkr0| \}}t|||}||f||fgS t | dkrNtg | |R  S t | dkrltg | |R  S tdd S r   )r   r#   r   r   r   )r   rf   r   r   midpointr0   r0   r1   _split_segment_at_t  s    r   MbP?c              	      sx  t | }t |}|sd}|s d}t||\}}|s6g S dd }	t| k rht| k rh|	||	|fgS t| d\}
}|d |	|f}|	||d f}t|d\}}|d |	|f}|	||d f}g }|t|
| ||d |t|| ||d |t|
| ||d |t|| ||d  fdd	}t }g }|D ]0}||}||v r\qB|| || qB|S )
N)r^   rA   c                 S   s   d| d | d   S )Nr3   r   rM   r0   )rr0   r0   r1   r   1  s    z._curve_curve_intersections_t.<locals>.midpointr3   r   rM   )range1range2c                    s    t | d   t | d   fS )Nr   rM   )int)r   	precisionr0   r1   <lambda>V  rv   z._curve_curve_intersections_t.<locals>.<lambda>)	r   r   r   r   extend_curve_curve_intersections_tsetaddrp   )curve1curve2r   r   r   Zbounds1Zbounds2Z
intersects_r   Zc11Zc12Z	c11_rangeZ	c12_rangeZc21Zc22Z	c21_rangeZ	c22_rangefoundZ
unique_keyseenZunique_valuesr   keyr0   r   r1   r   !  sb    





r   c                 C   s    t | | }tdd |D S )Nc                 s   s   | ]}t |d  dV  qdS )rM   r^   N)rN   r   )re   pr0   r0   r1   r~   f  rv   z_is_linelike.<locals>.<genexpr>)r   r   all)r   Z	maybeliner0   r0   r1   _is_lineliked  s    r   c                    s   t  rN d  d f}t |rB|d |d f}tg ||R  S t||S n"t |rp|d |d f}t |S t |} fdd|D S )a  Finds intersections between a curve and a curve.

    Args:
        curve1: List of coordinates of the first curve segment as 2D tuples.
        curve2: List of coordinates of the second curve segment as 2D tuples.

    Returns:
        A list of ``Intersection`` objects, each object having ``pt``, ``t1``
        and ``t2`` attributes containing the intersection point, time on first
        segment and time on second segment respectively.

    Examples::
        >>> curve1 = [ (10,100), (90,30), (40,140), (220,220) ]
        >>> curve2 = [ (5,150), (180,20), (80,250), (210,190) ]
        >>> intersections = curveCurveIntersections(curve1, curve2)
        >>> len(intersections)
        3
        >>> intersections[0].pt
        (81.7831487395506, 109.88904552375288)
    r   r   c                    s,   g | ]$}t t |d  |d  |d dqS )r   rM   r	   )r   r$   )re   r   r   r0   r1   rn     s   z+curveCurveIntersections.<locals>.<listcomp>)r   r%   r&   r   )r   r   Zline1Zline2Zintersection_tsr0   r   r1   r'   i  s    


r'   c                 C   s   d}t |t | kr"| | }} d}t | dkrRt |dkrFt| |}qt| |}n4t | dkr~t |dkr~tg | |R  }ntd|s|S dd |D S )a)  Finds intersections between two segments.

    Args:
        seg1: List of coordinates of the first segment as 2D tuples.
        seg2: List of coordinates of the second segment as 2D tuples.

    Returns:
        A list of ``Intersection`` objects, each object having ``pt``, ``t1``
        and ``t2`` attributes containing the intersection point, time on first
        segment and time on second segment respectively.

    Examples::
        >>> curve1 = [ (10,100), (90,30), (40,140), (220,220) ]
        >>> curve2 = [ (5,150), (180,20), (80,250), (210,190) ]
        >>> intersections = segmentSegmentIntersections(curve1, curve2)
        >>> len(intersections)
        3
        >>> intersections[0].pt
        (81.7831487395506, 109.88904552375288)
        >>> curve3 = [ (100, 240), (30, 60), (210, 230), (160, 30) ]
        >>> line  = [ (25, 260), (230, 20) ]
        >>> intersections = segmentSegmentIntersections(curve3, line)
        >>> len(intersections)
        3
        >>> intersections[0].pt
        (84.9000930760723, 189.87306176459828)

    FTrL   z4Couldn't work out which intersection function to usec                 S   s    g | ]}t |j|j|jd qS )r	   )r   r
   r   r   r   r0   r0   r1   rn     rv   z/segmentSegmentIntersections.<locals>.<listcomp>)r   r'   r&   r%   r   )Zseg1Zseg2Zswappedr   r0   r0   r1   r(     s    
r(   c                 C   sD   zt | }W n ty&   d|   Y S 0 dddd |D  S dS )zw
    >>> _segmentrepr([1, [2, 3], [], [[2, [3, 4], [0.1, 2.2]]]])
    '(1, (2, 3), (), ((2, (3, 4), (0.1, 2.2))))'
    z%gz(%s)z, c                 s   s   | ]}t |V  qd S rF   )_segmentrepr)re   rK   r0   r0   r1   r~     rv   z_segmentrepr.<locals>.<genexpr>N)iter	TypeErrorjoin)objitr0   r0   r1   r     s
    r   c                 C   s   | D ]}t t| qdS )zlHelper for the doctests, displaying each segment in a list of
    segments on a single line as a tuple.
    N)printr   )r   r   r0   r0   r1   printSegments  s    r   __main__)r)   )r)   )r   NN)X__doc__ZfontTools.misc.arrayToolsr   r   r   ZfontTools.misc.transformr   rN   collectionsr   r   AttributeErrorImportErrorZfontTools.miscZcompiledZCOMPILEDr>   r   __all__r   r8   returnsdoublelocalsr*   r?   r   r   r_   ZcfuncinlinerI   rQ   r   r   r   r   r   r   r   r   r   r   r   r   r   r   r   r   r   r   rO   r   r   r   r   r   ro   rz   r   r   r   r   r#   r    r!   r"   r$   r   r   r%   r   r   r&   r   r   r   r   r'   r(   r   r   __name__sysdoctestexittestmodfailedr0   r0   r0   r1   <module>   s   
	



#
		 
!"
$&9-%



#
!a		
N
& 
C'0
