U
    EvfEA                     @  sH  d Z ddlmZ ddlmZmZmZ ddlZddlm	Z	 ddl
mZmZ ddlmZmZ ddlmZmZmZmZmZmZmZmZmZ d	Zed
edZededZedZedZddddddZddddddZ G dd deeef Z!edddddZ"edd d!d"dZ"ed
ddd#d$dZ"d%d Z"d
ddd#d&d'Z#dS )(a  
Cycler
======

Cycling through combinations of values, producing dictionaries.

You can add cyclers::

    from cycler import cycler
    cc = (cycler(color=list('rgb')) +
          cycler(linestyle=['-', '--', '-.']))
    for d in cc:
        print(d)

Results in::

    {'color': 'r', 'linestyle': '-'}
    {'color': 'g', 'linestyle': '--'}
    {'color': 'b', 'linestyle': '-.'}


You can multiply cyclers::

    from cycler import cycler
    cc = (cycler(color=list('rgb')) *
          cycler(linestyle=['-', '--', '-.']))
    for d in cc:
        print(d)

Results in::

    {'color': 'r', 'linestyle': '-'}
    {'color': 'r', 'linestyle': '--'}
    {'color': 'r', 'linestyle': '-.'}
    {'color': 'g', 'linestyle': '-'}
    {'color': 'g', 'linestyle': '--'}
    {'color': 'g', 'linestyle': '-.'}
    {'color': 'b', 'linestyle': '-'}
    {'color': 'b', 'linestyle': '--'}
    {'color': 'b', 'linestyle': '-.'}
    )annotations)HashableIterable	GeneratorN)reduce)productcycle)muladd)	TypeVarGenericCallableUnionDictListAnyoverloadcastz0.12.1K)boundLVUz#Cycler[K, V] | Iterable[dict[K, V]]*Cycler[K, V] | Iterable[dict[K, V]] | Noneset[K])leftrightreturnc                 C  s`   | g krt t| ni }|dk	r,t t|ni }t| }t| }||@ rXtd||B S )a  
    Helper function to compose cycler keys.

    Parameters
    ----------
    left, right : iterable of dictionaries or None
        The cyclers to be composed.

    Returns
    -------
    keys : set
        The keys in the composition of the two cyclers.
    Nz"Can not compose overlapping cycles)nextitersetkeys
ValueError)r   r   Zl_peekZr_peekZl_keyZr_key r#   3/tmp/pip-unpacked-wheel-onpxacu0/cycler/__init__.py_process_keys>   s    r%   Cycler[K, V]zCycler[K, U]zCycler[K, V | U]c                   s   | j |j kr.tdj| j |j @ | j |j A dttttttt	f  f | 
  ttttttt	f  f |
 tt fdd| j D S )aS  
    Concatenate `Cycler`\s, as if chained using `itertools.chain`.

    The keys must match exactly.

    Examples
    --------
    >>> num = cycler('a', range(3))
    >>> let = cycler('a', 'abc')
    >>> num.concat(let)
    cycler('a', [0, 1, 2, 'a', 'b', 'c'])

    Returns
    -------
    `Cycler`
        The concatenated cycler.
    zBKeys do not match:
	Intersection: {both!r}
	Disjoint: {just_one!r})ZbothZjust_onec                 3  s$   | ]}t | | |  V  qd S N_cycler.0kZ_l_rr#   r$   	<genexpr>t   s     zconcat.<locals>.<genexpr>)r!   r"   formatr   r   r   r   r   r   r   by_keyr   r
   )r   r   r#   r-   r$   concatX   s    
 
""r2   c                   @  s~  e Zd ZdZdd ZdEddddd	d
Zdd ZeddddZddddddZ	e
ddddddZdddddZddd d!Zd"d#d$d%d&Zed"d#d$d'd(Zed)dd$d*d(Zd+d( Zed"d#d$d,d-Zed)dd$d.d-Zd/d- Zd)dd0d1Zddd$d2d3Zd4dd$d5d6Zd7d8d$d9d:ZdZd;dd<d=Zd;dd>d?Zd@ddAdBZeZdddCdDZeZdS )FCyclera  
    Composable cycles.

    This class has compositions methods:

    ``+``
      for 'inner' products (zip)

    ``+=``
      in-place ``+``

    ``*``
      for outer products (`itertools.product`) and integer multiplication

    ``*=``
      in-place ``*``

    and supports basic slicing via ``[]``.

    Parameters
    ----------
    left, right : Cycler or None
        The 'left' and 'right' cyclers.
    op : func or None
        Function which composes the 'left' and 'right' cyclers.
    c                 C  s   t | S r'   )r   selfr#   r#   r$   __call__   s    zCycler.__call__Nr   zCycler[K, V] | Noner   )r   r   opc                 C  s   t |tr t|j|j|j| _n |dk	r:dd |D | _ng | _t |tr`t|j|j|j| _nd| _t| j| j| _|| _dS )zf
        Semi-private init.

        Do not use this directly, use `cycler` function instead.
        Nc                 S  s   g | ]}t  |qS r#   )copyr+   vr#   r#   r$   
<listcomp>   s     z#Cycler.__init__.<locals>.<listcomp>)
isinstancer3   _left_right_opr%   _keys)r5   r   r   r7   r#   r#   r$   __init__   s$    
  
  zCycler.__init__c                 C  s
   || j kS r'   )r@   )r5   r,   r#   r#   r$   __contains__   s    zCycler.__contains__r   )r   c                 C  s
   t | jS )z!The keys this Cycler knows about.)r    r@   r4   r#   r#   r$   r!      s    zCycler.keysr   None)oldnewr   c                   s    krdS  | j kr2td d  d  d| j krXtd d  d d| j  | j   | jdk	r| jjkr| j  n4t| j	t
r| j	  n fdd| j	D | _	dS )	a  
        Change a key in this cycler to a new name.
        Modification is performed in-place.

        Does nothing if the old key is the same as the new key.
        Raises a ValueError if the new key is already a key.
        Raises a KeyError if the old key isn't a key.
        NzCan't replace z with , z is already a keyz is not a keyc                   s   g | ]} | iqS r#   r#   )r+   entryrE   rD   r#   r$   r;      s     z%Cycler.change_key.<locals>.<listcomp>)r@   r"   KeyErrorremover
   r>   r!   
change_keyr<   r=   r3   )r5   rD   rE   r#   rH   r$   rK      s"    	

zCycler.change_keyIterable[V]r&   labelitrr   c                   s,   | d}t  fdd|D |_ h|_|S )a  
        Class method to create 'base' Cycler objects
        that do not have a 'right' or 'op' and for which
        the 'left' object is not another Cycler.

        Parameters
        ----------
        label : hashable
            The property key.

        itr : iterable
            Finite length iterable of the property values.

        Returns
        -------
        `Cycler`
            New 'base' cycler.
        Nc                 3  s   | ]} |iV  qd S r'   r#   r9   rN   r#   r$   r/      s     z$Cycler._from_iter.<locals>.<genexpr>)listr=   r@   )clsrN   rO   retr#   rP   r$   
_from_iter   s    zCycler._from_iterslice)keyr   c                   s:   t  tr.|  }tt fdd| D S tdd S )Nc                 3  s    | ]\}}t ||  V  qd S r'   r(   r+   r,   r:   rV   r#   r$   r/      s     z%Cycler.__getitem__.<locals>.<genexpr>z+Can only use slices with Cycler.__getitem__)r<   rU   r1   r   r
   itemsr"   )r5   rV   transr#   rX   r$   __getitem__   s    
zCycler.__getitem__z!Generator[dict[K, V], None, None]c                 c  sp   | j d kr"| jD ]}t|V  qnJ| jd kr4td| | j| j D ]&\}}i }|| || |V  qDd S )Nz=Operation cannot be None when both left and right are defined)r>   r=   dictr?   	TypeErrorupdate)r5   r   aboutr#   r#   r$   __iter__  s    




zCycler.__iter__zCycler[L, U]zCycler[K | L, V | U])otherr   c                 C  st   t | t |kr,tdt |  dt | ttttttf tttf f | tttttf tttf f |t	S )z
        Pair-wise combine two equal length cyclers (zip).

        Parameters
        ----------
        other : Cycler
        z&Can only add equal length cycles, not z and )
lenr"   r3   r   r   r   r   r   r   zipr5   rc   r#   r#   r$   __add__  s      zCycler.__add__c                 C  s   d S r'   r#   rf   r#   r#   r$   __mul__$  s    zCycler.__mul__intc                 C  s   d S r'   r#   rf   r#   r#   r$   rh   (  s    c                   s   t  trRttttttf tttf f | tttttf tttf f  tS t  t	r| 
 }tt fdd| D S tS dS )z
        Outer product of two cyclers (`itertools.product`) or integer
        multiplication.

        Parameters
        ----------
        other : Cycler or int
        c                 3  s    | ]\}}t ||  V  qd S r'   r(   rW   rc   r#   r$   r/   >  s     z!Cycler.__mul__.<locals>.<genexpr>N)r<   r3   r   r   r   r   r   r   r   ri   r1   r   r
   rY   NotImplemented)r5   rc   rZ   r#   rj   r$   rh   ,  s    	
  
 c                 C  s   d S r'   r#   rf   r#   r#   r$   __rmul__C  s    zCycler.__rmul__c                 C  s   d S r'   r#   rf   r#   r#   r$   rl   G  s    c                 C  s   | | S r'   r#   rf   r#   r#   r$   rl   K  s    c                 C  sD   t ttti}| jd kr t| jS t| j}t| j}|| j ||S r'   )re   minr   r	   r>   rd   r=   r?   )r5   Zop_dictZl_lenZr_lenr#   r#   r$   __len__N  s    



zCycler.__len__c                 C  sL   t |tstdt| }t||| _|| _t| _t|j|j	|j| _	| S )z
        In-place pair-wise combine two equal length cyclers (zip).

        Parameters
        ----------
        other : Cycler
        z"Cannot += with a non-Cycler object)
r<   r3   r]   r8   r%   r@   r=   re   r?   r>   r5   rc   Zold_selfr#   r#   r$   __iadd__X  s    

zCycler.__iadd__zCycler[K, V] | intc                 C  sL   t |tstdt| }t||| _|| _t| _t|j|j	|j| _	| S )z
        In-place outer product of two cyclers (`itertools.product`).

        Parameters
        ----------
        other : Cycler
        z"Cannot *= with a non-Cycler object)
r<   r3   r]   r8   r%   r@   r=   r   r?   r>   ro   r#   r#   r$   __imul__j  s    

zCycler.__imul__objectboolc                 C  sJ   t |tsdS t| t|kr"dS | j|jA r2dS tdd t| |D S )NFc                 s  s   | ]\}}||kV  qd S r'   r#   )r+   r_   r`   r#   r#   r$   r/     s     z Cycler.__eq__.<locals>.<genexpr>)r<   r3   rd   r!   allre   rf   r#   r#   r$   __eq__|  s    
zCycler.__eq__strc                   sr   t dtdi}| jd krH| j  t fdd| D }d d|dS || jd}d	}|j| j	|| jd
S d S )N+*c                 3  s   | ]}|  V  qd S r'   r#   r9   Zlabr#   r$   r/     s     z"Cycler.__repr__.<locals>.<genexpr>zcycler(rF   )?z({left!r} {op} {right!r}))r   r7   r   )
re   r   r>   r!   poprQ   getr?   r0   r=   )r5   Zop_maprO   r7   msgr#   ry   r$   __repr__  s    

zCycler.__repr__c                 C  st   d}t | jtd}|D ]}|d|d7 }qt| D ]2}|d7 }|D ]}|d|| d7 }qD|d7 }q4|d	7 }|S )
Nz<table>rX   z<th>z</th>z<tr>z<td>z</td>z</tr>z</table>)sortedr!   reprr   )r5   outputZsorted_keysrV   dr,   r#   r#   r$   _repr_html_  s    
zCycler._repr_html_zdict[K, list[V]]c                 C  s>   | j }dd |D }| D ] }|D ]}|| ||  q q|S )a  
        Values by key.

        This returns the transposed values of the cycler.  Iterating
        over a `Cycler` yields dicts with a single value for each key,
        this method returns a `dict` of `list` which are the values
        for the given key.

        The returned value can be used to create an equivalent `Cycler`
        using only `+`.

        Returns
        -------
        transpose : dict
            dict of lists of the values for each key.
        c                 S  s   i | ]}|t  qS r#   )rQ   r*   r#   r#   r$   
<dictcomp>  s      z!Cycler.by_key.<locals>.<dictcomp>)r!   append)r5   r!   ra   r   r,   r#   r#   r$   r1     s    zCycler.by_keyc                 C  s    |   }ttdd | D S )z
        Simplify the cycler into a sum (but no products) of cyclers.

        Returns
        -------
        simple : Cycler
        c                 s  s   | ]\}}t ||V  qd S r'   r(   rW   r#   r#   r$   r/     s     z"Cycler.simplify.<locals>.<genexpr>)r1   r   r
   rY   )r5   rZ   r#   r#   r$   simplify  s    zCycler.simplify)NN)__name__
__module____qualname____doc__r6   rA   rB   propertyr!   rK   classmethodrT   r[   rb   rg   r   rh   rl   rn   rp   rq   ru   __hash__r   r   r1   Z
_transposer   r2   r#   r#   r#   r$   r3   w   sF      $
	r3   )argr   c                 C  s   d S r'   r#   )r   r#   r#   r$   cycler  s    r   rL   zCycler[str, V])kwargsr   c                  K  s   d S r'   r#   )r   r#   r#   r$   r     s    rM   c                 C  s   d S r'   r#   )rN   rO   r#   r#   r$   r     s    c                  O  s   | r|rt dt| dkr>t| d ts2t dt| d S t| dkrRt|  S t| dkrft d|rttdd | D S t d	d
S )a  
    Create a new `Cycler` object from a single positional argument,
    a pair of positional arguments, or the combination of keyword arguments.

    cycler(arg)
    cycler(label1=itr1[, label2=iter2[, ...]])
    cycler(label, itr)

    Form 1 simply copies a given `Cycler` object.

    Form 2 composes a `Cycler` as an inner product of the
    pairs of keyword arguments. In other words, all of the
    iterables are cycled simultaneously, as if through zip().

    Form 3 creates a `Cycler` from a label and an iterable.
    This is useful for when the label cannot be a keyword argument
    (e.g., an integer or a name that has a space in it).

    Parameters
    ----------
    arg : Cycler
        Copy constructor for Cycler (does a shallow copy of iterables).
    label : name
        The property key. In the 2-arg form of the function,
        the label can be any hashable object. In the keyword argument
        form of the function, it must be a valid python identifier.
    itr : iterable
        Finite length iterable of the property values.
        Can be a single-property `Cycler` that would
        be like a key change, but as a shallow copy.

    Returns
    -------
    cycler : Cycler
        New `Cycler` for the given property

    zEcycler() can only accept positional OR keyword arguments -- not both.   r   zDIf only one positional argument given, it must be a Cycler instance.   zdOnly a single Cycler can be accepted as the lone positional argument. Use keyword arguments instead.c                 s  s   | ]\}}t ||V  qd S r'   r(   rW   r#   r#   r$   r/     s     zcycler.<locals>.<genexpr>z4Must have at least a positional OR keyword argumentsN)r]   rd   r<   r3   r)   r   r
   rY   )argsr   r#   r#   r$   r     s&    &c                   sN   t |trB|j}t|dkr(d}t||   fdd|D }t| |S )aD  
    Create a new `Cycler` object from a property name and iterable of values.

    Parameters
    ----------
    label : hashable
        The property key.
    itr : iterable
        Finite length iterable of the property values.

    Returns
    -------
    cycler : Cycler
        New `Cycler` for the given property
    r   z2Can not create Cycler from a multi-property Cyclerc                 3  s   | ]}|  V  qd S r'   r#   r9   ry   r#   r$   r/   ;  s     z_cycler.<locals>.<genexpr>)r<   r3   r!   rd   r"   r|   rT   )rN   rO   r!   r~   r#   ry   r$   r)   "  s    
r)   )$r   
__future__r   collections.abcr   r   r   r8   	functoolsr   	itertoolsr   r   operatorr	   r
   typingr   r   r   r   r   r   r   r   r   __version__r   r   r   r   r%   r2   r3   r   r)   r#   r#   r#   r$   <module>   s2   +,  ^@