o
    _h0                     @   s   d Z ddlmZmZ ddlmZmZ g dZdd ZG dd deZ	G d	d
 d
ee	Z
G dd deZG dd deeZdddZedkrje	 Zed ed eddd e  ddlmZ eej dS dS )z:Pen recording operations that can be accessed or replayed.    )AbstractPenDecomposingPen)AbstractPointPenDecomposingPointPen)replayRecordingRecordingPenDecomposingRecordingPenDecomposingRecordingPointPenRecordingPointPenlerpRecordingsc                 C   s    | D ]\}}t |||  qdS )a2  Replay a recording, as produced by RecordingPen or DecomposingRecordingPen,
    to a pen.

    Note that recording does not have to be produced by those pens.
    It can be any iterable of tuples of method name and tuple-of-arguments.
    Likewise, pen can be any objects receiving those method calls.
    N)getattr)	recordingpenoperatoroperands r   O/var/www/html/myenv/lib/python3.10/site-packages/fontTools/pens/recordingPen.pyr      s   r   c                   @   sd   e Zd ZdZdd Zdd Zdd Zdd	 Zd
d Zdd Z	dd Z
dd Zdd Zdd ZeZdS )r   aP  Pen recording operations that can be accessed or replayed.

    The recording can be accessed as pen.value; or replayed using
    pen.replay(otherPen).

    :Example:
        .. code-block::

            from fontTools.ttLib import TTFont
            from fontTools.pens.recordingPen import RecordingPen

            glyph_name = 'dollar'
            font_path = 'MyFont.otf'

            font = TTFont(font_path)
            glyphset = font.getGlyphSet()
            glyph = glyphset[glyph_name]

            pen = RecordingPen()
            glyph.draw(pen)
            print(pen.value)
    c                 C   
   g | _ d S Nvalueselfr   r   r   __init__5      
zRecordingPen.__init__c                 C      | j d|ff d S )NmoveTor   append)r   p0r   r   r   r   8      zRecordingPen.moveToc                 C   r   )NlineTor   )r   p1r   r   r   r!   ;   r    zRecordingPen.lineToc                 G      | j d|f d S )NqCurveTor   r   pointsr   r   r   r$   >      zRecordingPen.qCurveToc                 G   r#   )NcurveTor   r%   r   r   r   r(   A   r'   zRecordingPen.curveToc                 C      | j d d S )N)	closePathr   r   r   r   r   r   r*   D      zRecordingPen.closePathc                 C   r)   )N)endPathr   r   r   r   r   r   r,   G   r+   zRecordingPen.endPathc                 C   s   | j d||ff d S )NaddComponentr   )r   	glyphNametransformationr   r   r   r-   J   s   zRecordingPen.addComponentc                 C   s   | j d|||ff d S )NaddVarComponentr   )r   r.   r/   locationr   r   r   r0   M   s   zRecordingPen.addVarComponentc                 C   s   t | j| d S r   )r   r   )r   r   r   r   r   replayP   r+   zRecordingPen.replayN)__name__
__module____qualname____doc__r   r   r!   r$   r(   r*   r,   r-   r0   r2   drawr   r   r   r   r      s    r   c                   @      e Zd ZdZdZdS )r   aa	  Same as RecordingPen, except that it doesn't keep components
    as references, but draws them decomposed as regular contours.

    The constructor takes a required 'glyphSet' positional argument,
    a dictionary of glyph objects (i.e. with a 'draw' method) keyed
    by thir name; other arguments are forwarded to the DecomposingPen's
    constructor::

        >>> class SimpleGlyph(object):
        ...     def draw(self, pen):
        ...         pen.moveTo((0, 0))
        ...         pen.curveTo((1, 1), (2, 2), (3, 3))
        ...         pen.closePath()
        >>> class CompositeGlyph(object):
        ...     def draw(self, pen):
        ...         pen.addComponent('a', (1, 0, 0, 1, -1, 1))
        >>> class MissingComponent(object):
        ...     def draw(self, pen):
        ...         pen.addComponent('foobar', (1, 0, 0, 1, 0, 0))
        >>> class FlippedComponent(object):
        ...     def draw(self, pen):
        ...         pen.addComponent('a', (-1, 0, 0, 1, 0, 0))
        >>> glyphSet = {
        ...    'a': SimpleGlyph(),
        ...    'b': CompositeGlyph(),
        ...    'c': MissingComponent(),
        ...    'd': FlippedComponent(),
        ... }
        >>> for name, glyph in sorted(glyphSet.items()):
        ...     pen = DecomposingRecordingPen(glyphSet)
        ...     try:
        ...         glyph.draw(pen)
        ...     except pen.MissingComponentError:
        ...         pass
        ...     print("{}: {}".format(name, pen.value))
        a: [('moveTo', ((0, 0),)), ('curveTo', ((1, 1), (2, 2), (3, 3))), ('closePath', ())]
        b: [('moveTo', ((-1, 1),)), ('curveTo', ((0, 2), (1, 3), (2, 4))), ('closePath', ())]
        c: []
        d: [('moveTo', ((0, 0),)), ('curveTo', ((-1, 1), (-2, 2), (-3, 3))), ('closePath', ())]

        >>> for name, glyph in sorted(glyphSet.items()):
        ...     pen = DecomposingRecordingPen(
        ...         glyphSet, skipMissingComponents=True, reverseFlipped=True,
        ...     )
        ...     glyph.draw(pen)
        ...     print("{}: {}".format(name, pen.value))
        a: [('moveTo', ((0, 0),)), ('curveTo', ((1, 1), (2, 2), (3, 3))), ('closePath', ())]
        b: [('moveTo', ((-1, 1),)), ('curveTo', ((0, 2), (1, 3), (2, 4))), ('closePath', ())]
        c: []
        d: [('moveTo', ((0, 0),)), ('lineTo', ((-3, 3),)), ('curveTo', ((-2, 2), (-1, 1), (0, 0))), ('closePath', ())]
    FNr3   r4   r5   r6   skipMissingComponentsr   r   r   r   r   V   s    5r   c                   @   sX   e Zd ZdZdd ZdddZdd Z	dd
dZdddZ	dddZ	dd Z
e
ZdS )r
   a  PointPen recording operations that can be accessed or replayed.

    The recording can be accessed as pen.value; or replayed using
    pointPen.replay(otherPointPen).

    :Example:
        .. code-block::

            from defcon import Font
            from fontTools.pens.recordingPen import RecordingPointPen

            glyph_name = 'a'
            font_path = 'MyFont.ufo'

            font = Font(font_path)
            glyph = font[glyph_name]

            pen = RecordingPointPen()
            glyph.drawPoints(pen)
            print(pen.value)

            new_glyph = font.newGlyph('b')
            pen.replay(new_glyph.getPointPen())
    c                 C   r   r   r   r   r   r   r   r      r   zRecordingPointPen.__init__Nc                 K   s&   |d ur||d< | j dd|f d S )N
identifier	beginPathr   r   )r   r;   kwargsr   r   r   r<      s   zRecordingPointPen.beginPathc                 C   s   | j ddi f d S )Nr,   r   r   r   r   r   r   r,      r    zRecordingPointPen.endPathFc                 K   s.   |d ur||d< | j d||||f|f d S )Nr;   addPointr   )r   ptsegmentTypesmoothnamer;   r=   r   r   r   r>      s   zRecordingPointPen.addPointc                 K   s*   |d ur||d< | j d||f|f d S )Nr;   r-   r   )r   baseGlyphNamer/   r;   r=   r   r   r   r-      s   zRecordingPointPen.addComponentc                 K   s,   |d ur||d< | j d|||f|f d S )Nr;   r0   r   )r   rC   r/   r1   r;   r=   r   r   r   r0      s
   z!RecordingPointPen.addVarComponentc                 C   s*   | j D ]\}}}t|||i | qd S r   )r   r   )r   pointPenr   argsr=   r   r   r   r2      s   zRecordingPointPen.replayr   )NFNN)r3   r4   r5   r6   r   r<   r,   r>   r-   r0   r2   
drawPointsr   r   r   r   r
      s    



	r
   c                   @   r8   )r	   a  Same as RecordingPointPen, except that it doesn't keep components
    as references, but draws them decomposed as regular contours.

    The constructor takes a required 'glyphSet' positional argument,
    a dictionary of pointPen-drawable glyph objects (i.e. with a 'drawPoints' method)
    keyed by thir name; other arguments are forwarded to the DecomposingPointPen's
    constructor::

        >>> from pprint import pprint
        >>> class SimpleGlyph(object):
        ...     def drawPoints(self, pen):
        ...         pen.beginPath()
        ...         pen.addPoint((0, 0), "line")
        ...         pen.addPoint((1, 1))
        ...         pen.addPoint((2, 2))
        ...         pen.addPoint((3, 3), "curve")
        ...         pen.endPath()
        >>> class CompositeGlyph(object):
        ...     def drawPoints(self, pen):
        ...         pen.addComponent('a', (1, 0, 0, 1, -1, 1))
        >>> class MissingComponent(object):
        ...     def drawPoints(self, pen):
        ...         pen.addComponent('foobar', (1, 0, 0, 1, 0, 0))
        >>> class FlippedComponent(object):
        ...     def drawPoints(self, pen):
        ...         pen.addComponent('a', (-1, 0, 0, 1, 0, 0))
        >>> glyphSet = {
        ...    'a': SimpleGlyph(),
        ...    'b': CompositeGlyph(),
        ...    'c': MissingComponent(),
        ...    'd': FlippedComponent(),
        ... }
        >>> for name, glyph in sorted(glyphSet.items()):
        ...     pen = DecomposingRecordingPointPen(glyphSet)
        ...     try:
        ...         glyph.drawPoints(pen)
        ...     except pen.MissingComponentError:
        ...         pass
        ...     pprint({name: pen.value})
        {'a': [('beginPath', (), {}),
               ('addPoint', ((0, 0), 'line', False, None), {}),
               ('addPoint', ((1, 1), None, False, None), {}),
               ('addPoint', ((2, 2), None, False, None), {}),
               ('addPoint', ((3, 3), 'curve', False, None), {}),
               ('endPath', (), {})]}
        {'b': [('beginPath', (), {}),
               ('addPoint', ((-1, 1), 'line', False, None), {}),
               ('addPoint', ((0, 2), None, False, None), {}),
               ('addPoint', ((1, 3), None, False, None), {}),
               ('addPoint', ((2, 4), 'curve', False, None), {}),
               ('endPath', (), {})]}
        {'c': []}
        {'d': [('beginPath', (), {}),
               ('addPoint', ((0, 0), 'line', False, None), {}),
               ('addPoint', ((-1, 1), None, False, None), {}),
               ('addPoint', ((-2, 2), None, False, None), {}),
               ('addPoint', ((-3, 3), 'curve', False, None), {}),
               ('endPath', (), {})]}

        >>> for name, glyph in sorted(glyphSet.items()):
        ...     pen = DecomposingRecordingPointPen(
        ...         glyphSet, skipMissingComponents=True, reverseFlipped=True,
        ...     )
        ...     glyph.drawPoints(pen)
        ...     pprint({name: pen.value})
        {'a': [('beginPath', (), {}),
               ('addPoint', ((0, 0), 'line', False, None), {}),
               ('addPoint', ((1, 1), None, False, None), {}),
               ('addPoint', ((2, 2), None, False, None), {}),
               ('addPoint', ((3, 3), 'curve', False, None), {}),
               ('endPath', (), {})]}
        {'b': [('beginPath', (), {}),
               ('addPoint', ((-1, 1), 'line', False, None), {}),
               ('addPoint', ((0, 2), None, False, None), {}),
               ('addPoint', ((1, 3), None, False, None), {}),
               ('addPoint', ((2, 4), 'curve', False, None), {}),
               ('endPath', (), {})]}
        {'c': []}
        {'d': [('beginPath', (), {}),
               ('addPoint', ((0, 0), 'curve', False, None), {}),
               ('addPoint', ((-3, 3), 'line', False, None), {}),
               ('addPoint', ((-2, 2), None, False, None), {}),
               ('addPoint', ((-1, 1), None, False, None), {}),
               ('endPath', (), {})]}
    FNr9   r   r   r   r   r	      s    Wr	         ?c                 #   s    t | t |krtdt | t |f t| |D ]-\\}}\}}||kr.td||f |dkr6td fddt||D }||fV  qdS )a  Linearly interpolate between two recordings. The recordings
    must be decomposed, i.e. they must not contain any components.

    Factor is typically between 0 and 1. 0 means the first recording,
    1 means the second recording, and 0.5 means the average of the
    two recordings. Other values are possible, and can be useful to
    extrapolate. Defaults to 0.5.

    Returns a generator with the new recording.
    zMismatched lengths: %d and %dzMismatched operations: %s, %sr-   zCannot interpolate componentsc                    s8   g | ]\\}}\}}|||    |||    fqS r   r   ).0x1y1x2y2factorr   r   
<listcomp>@  s    z"lerpRecordings.<locals>.<listcomp>N)len
ValueErrorzip)
recording1
recording2rN   op1args1op2args2mid_argsr   rM   r   r   +  s   
r   __main__)r   r   )r   d   )2   K   )<   r\   )r\      )pprintN)rG   )r6   fontTools.pens.basePenr   r   fontTools.pens.pointPenr   r   __all__r   r   r   r
   r	   r   r3   r   r   r!   r(   r*   r`   r   r   r   r   r   <module>   s&    
99A
[

