U
    b!I                     @  s   d dl mZ d dlZd dlmZmZ d dlZd dlm	Z	m
Z
mZmZ d dlmZ d dlmZmZ d dlmZmZ erd dlmZ G d	d
 d
eZG dd deZdS )    )annotationsN)TYPE_CHECKINGAny)FilePath
ReadBufferScalarStorageOptions)import_optional_dependency)BaseExcelReaderExcelWriter)combine_kwargsvalidate_freeze_panes)Serialisablec                      s   e Zd ZdZdZd(ddddd	 fd
dZdd ZedddddZedd Z	edd Z
edd Zedd Zedd Zedd Zedd  Zed!d" Zed#d$ Zd)d&d'Z  ZS )*OpenpyxlWriteropenpyxl)z.xlsxz.xlsmNwstrr   z
str | Nonezdict[str, Any] | Nonemodestorage_optionsif_sheet_existsengine_kwargsc	                   s   ddl m}
 t||	}t j|||||d d jkr|ddlm} | jj	f| _
 jj	d  fdd j
jD  _n(|
f | _
 j
jr j
 j
jd  d S )Nr   Workbookr   r+load_workbookc                   s   i | ]}| j | qS  )book).0nameselfr   =/tmp/pip-unpacked-wheel-ck39h295/pandas/io/excel/_openpyxl.py
<dictcomp>D   s      z+OpenpyxlWriter.__init__.<locals>.<dictcomp>)Zopenpyxl.workbookr   r   super__init__r   r   r   handleshandler   seekZ
sheetnamessheets
worksheetsremove)r"   pathengineZdate_formatZdatetime_formatr   r   r   r   kwargsr   r   	__class__r!   r#   r&   $   s"    


zOpenpyxlWriter.__init__c                 C  s:   | j | jj d| jkr6t| jjtjs6| jj  dS )z(
        Save workbook to disk.
        r   N)r   saver'   r(   r   
isinstancemmaptruncater!   r   r   r#   r2   M   s    zOpenpyxlWriter.savedictzdict[str, Serialisable])
style_dictreturnc                 C  s\   ddi}i }|  D ]B\}}||kr,|| }t| d| dd }||}|r|||< q|S )a  
        Convert a style_dict to a set of kwargs suitable for initializing
        or updating-on-copy an openpyxl v2 style object.

        Parameters
        ----------
        style_dict : dict
            A dict with zero or more of the following keys (or their synonyms).
                'font'
                'fill'
                'border' ('borders')
                'alignment'
                'number_format'
                'protection'

        Returns
        -------
        style_kwargs : dict
            A dict with the same, normalized keys as ``style_dict`` but each
            value has been replaced with a native openpyxl style object of the
            appropriate class.
        ZbordersZborderZ_convert_to_c                 S  s   d S Nr   )xr   r   r#   <lambda>t       z9OpenpyxlWriter._convert_to_style_kwargs.<locals>.<lambda>)itemsgetattr)clsr7   Z_style_key_mapstyle_kwargskvZ
_conv_to_xZnew_vr   r   r#   _convert_to_style_kwargsV   s    
z'OpenpyxlWriter._convert_to_style_kwargsc                 C  s,   ddl m} t|tr||S |f |S dS )a  
        Convert ``color_spec`` to an openpyxl v2 Color object.

        Parameters
        ----------
        color_spec : str, dict
            A 32-bit ARGB hex string, or a dict with zero or more of the
            following keys.
                'rgb'
                'indexed'
                'auto'
                'theme'
                'tint'
                'index'
                'type'

        Returns
        -------
        color : openpyxl.styles.Color
        r   )ColorN)openpyxl.stylesrD   r3   r   )r?   Z
color_specrD   r   r   r#   _convert_to_color{   s    
z OpenpyxlWriter._convert_to_colorc                 C  sh   ddl m} ddddddd	}i }| D ]2\}}||krB|| }|d
krT| |}|||< q*|f |S )a  
        Convert ``font_dict`` to an openpyxl v2 Font object.

        Parameters
        ----------
        font_dict : dict
            A dict with zero or more of the following keys (or their synonyms).
                'name'
                'size' ('sz')
                'bold' ('b')
                'italic' ('i')
                'underline' ('u')
                'strikethrough' ('strike')
                'color'
                'vertAlign' ('vertalign')
                'charset'
                'scheme'
                'family'
                'outline'
                'shadow'
                'condense'

        Returns
        -------
        font : openpyxl.styles.Font
        r   )FontsizeZboldZitalicZ	underlineZstrikethroughZ	vertAlign)szbiustrikeZ	vertaligncolor)rE   rG   r=   rF   )r?   Z	font_dictrG   Z_font_key_mapZfont_kwargsrA   rB   r   r   r#   _convert_to_font   s     	

zOpenpyxlWriter._convert_to_fontc                 C  s   t | j|S )a  
        Convert ``stop_seq`` to a list of openpyxl v2 Color objects,
        suitable for initializing the ``GradientFill`` ``stop`` parameter.

        Parameters
        ----------
        stop_seq : iterable
            An iterable that yields objects suitable for consumption by
            ``_convert_to_color``.

        Returns
        -------
        stop : list of openpyxl.styles.Color
        )maprF   )r?   Zstop_seqr   r   r#   _convert_to_stop   s    zOpenpyxlWriter._convert_to_stopc                 C  s   ddl m}m} ddddddd}ddi}i }i }| D ]\}}	d }
}||krZ|| }
||krj|| }|
d	kr|| |	}	|d
kr| |	}	|
r|	||
< q:|r|	||< q:|	||< |	||< q:z|f |W S  tk
r   |f | Y S X dS )a  
        Convert ``fill_dict`` to an openpyxl v2 Fill object.

        Parameters
        ----------
        fill_dict : dict
            A dict with one or more of the following keys (or their synonyms),
                'fill_type' ('patternType', 'patterntype')
                'start_color' ('fgColor', 'fgcolor')
                'end_color' ('bgColor', 'bgcolor')
            or one or more of the following keys (or their synonyms).
                'type' ('fill_type')
                'degree'
                'left'
                'right'
                'top'
                'bottom'
                'stop'

        Returns
        -------
        fill : openpyxl.styles.Fill
        r   )GradientFillPatternFillZ	fill_typestart_color	end_color)ZpatternTypeZpatterntypeZfgColorZfgcolorZbgColorZbgcolortypeN)rT   rU   stop)rE   rR   rS   r=   rF   rQ   	TypeError)r?   Z	fill_dictrR   rS   Z_pattern_fill_key_mapZ_gradient_fill_key_mapZpfill_kwargsZgfill_kwargsrA   rB   pkZgkr   r   r#   _convert_to_fill   s>    	




zOpenpyxlWriter._convert_to_fillc                 C  sr   ddl m} ddi}t|tr(||dS i }| D ]2\}}||krL|| }|dkr^| |}|||< q4|f |S )a  
        Convert ``side_spec`` to an openpyxl v2 Side object.

        Parameters
        ----------
        side_spec : str, dict
            A string specifying the border style, or a dict with zero or more
            of the following keys (or their synonyms).
                'style' ('border_style')
                'color'

        Returns
        -------
        side : openpyxl.styles.Side
        r   )SideZborder_stylestyle)r\   rN   )rE   r[   r3   r   r=   rF   )r?   Z	side_specr[   Z_side_key_mapZside_kwargsrA   rB   r   r   r#   _convert_to_side  s    



zOpenpyxlWriter._convert_to_sidec                 C  sr   ddl m} ddd}i }| D ]D\}}||kr:|| }|dkrL| |}|dkr^| |}|||< q"|f |S )aq  
        Convert ``border_dict`` to an openpyxl v2 Border object.

        Parameters
        ----------
        border_dict : dict
            A dict with zero or more of the following keys (or their synonyms).
                'left'
                'right'
                'top'
                'bottom'
                'diagonal'
                'diagonal_direction'
                'vertical'
                'horizontal'
                'diagonalUp' ('diagonalup')
                'diagonalDown' ('diagonaldown')
                'outline'

        Returns
        -------
        border : openpyxl.styles.Border
        r   )BorderZ
diagonalUpZdiagonalDown)Z
diagonalupZdiagonaldownrN   )leftrighttopZbottomZdiagonal)rE   r^   r=   rF   r]   )r?   Zborder_dictr^   Z_border_key_mapZborder_kwargsrA   rB   r   r   r#   _convert_to_border?  s    



z!OpenpyxlWriter._convert_to_borderc                 C  s   ddl m} |f |S )a  
        Convert ``alignment_dict`` to an openpyxl v2 Alignment object.

        Parameters
        ----------
        alignment_dict : dict
            A dict with zero or more of the following keys (or their synonyms).
                'horizontal'
                'vertical'
                'text_rotation'
                'wrap_text'
                'shrink_to_fit'
                'indent'
        Returns
        -------
        alignment : openpyxl.styles.Alignment
        r   )	Alignment)rE   rc   )r?   Zalignment_dictrc   r   r   r#   _convert_to_alignmenth  s    z$OpenpyxlWriter._convert_to_alignmentc                 C  s   |d S )aO  
        Convert ``number_format_dict`` to an openpyxl v2.1.0 number format
        initializer.

        Parameters
        ----------
        number_format_dict : dict
            A dict with zero or more of the following keys.
                'format_code' : str

        Returns
        -------
        number_format : str
        Zformat_coder   )r?   Znumber_format_dictr   r   r#   _convert_to_number_format  s    z(OpenpyxlWriter._convert_to_number_formatc                 C  s   ddl m} |f |S )a'  
        Convert ``protection_dict`` to an openpyxl v2 Protection object.

        Parameters
        ----------
        protection_dict : dict
            A dict with zero or more of the following keys.
                'locked'
                'hidden'

        Returns
        -------
        r   )
Protection)rE   rf   )r?   Zprotection_dictrf   r   r   r#   _convert_to_protection  s    z%OpenpyxlWriter._convert_to_protectionr   c                 C  s  |  |}i }|| jkr| jdkrd| jkr| jdkrn| j| }| j|}| j|= | j||}	|	| j|< q| jdkrtd| dq| jdkr| j| }	qtd| j d	q| j| }	n| j }	||	_|	| j|< t	|r|	j
|d
 d |d d d|	_|D ]}
|	j
||
j d ||
j d d}| |
j\|_}|rN||_i }|
jrt|
j}||}|d kr| |
j}|||< |r| D ]\}}t||| q|
jd k	r|
jd k	r|	j||
j d ||
j d ||
j d ||
j d d |r||
j d }||
j d }||
j d }||
j d }t||d D ]b}t||d D ]L}||kr~||kr~qb|	j
||d}| D ]\}}t||| qqbqPqd S )Nnewr   replaceerrorzSheet 'z7' already exists and if_sheet_exists is set to 'error'.overlay'z^' is not valid for if_sheet_exists. Valid options are 'error', 'new', 'replace' and 'overlay'.r      )rowcolumn)Z	start_rowZstart_columnZ
end_columnZend_row)ro   rn   )Z_get_sheet_namer*   r   r   r   indexZcreate_sheet
ValueErrortitler   cellfreeze_panesrn   colZ_value_with_fmtvalvalueZnumber_formatr\   r   getrC   r=   setattrZ
mergestartZmergeendZmerge_cellsrange)r"   cellsZ
sheet_nameZstartrowZstartcolrt   Z_style_cacheZold_wksZtarget_indexZwksrs   Zxcellfmtr@   keyrA   rB   Z	first_rowZlast_rowZ	first_colZlast_colrn   ru   r   r   r#   write_cells  s    










 

 



zOpenpyxlWriter.write_cells)NNNr   NNN)Nr   r   N)__name__
__module____qualname__r.   Zsupported_extensionsr&   r2   classmethodrC   rF   rO   rQ   rZ   r]   rb   rd   re   rg   r~   __classcell__r   r   r0   r#   r       sH          )	$

0

A
!
(


       r   c                      s   e Zd Zd!dddd fddZedd	 Zdd
ddZeddddZddddZddddZ	dddddZ
ddddd Z  ZS )"OpenpyxlReaderNzFilePath | ReadBuffer[bytes]r   None)filepath_or_bufferr   r8   c                   s   t d t j||d dS )a.  
        Reader using openpyxl engine.

        Parameters
        ----------
        filepath_or_buffer : str, path object or Workbook
            Object to be parsed.
        storage_options : dict, optional
            passed to fsspec for appropriate URLs (see ``_get_filepath_or_buffer``)
        r   )r   N)r	   r%   r&   )r"   r   r   r0   r   r#   r&     s    zOpenpyxlReader.__init__c                 C  s   ddl m} |S )Nr   r   )r   r   )r"   r   r   r   r#   _workbook_class  s    zOpenpyxlReader._workbook_class)r   c                 C  s   ddl m} ||ddddS )Nr   r   TF)	read_onlyZ	data_onlyZ
keep_links)r   r   )r"   r   r   r   r   r#   r     s       zOpenpyxlReader.load_workbookz	list[str])r8   c                 C  s   dd | j jD S )Nc                 S  s   g | ]
}|j qS r   )rr   )r   sheetr   r   r#   
<listcomp>  s     z.OpenpyxlReader.sheet_names.<locals>.<listcomp>)r   r+   r!   r   r   r#   sheet_names  s    zOpenpyxlReader.sheet_namesr   )r    c                 C  s   |  | | j| S r9   )Zraise_if_bad_sheet_by_namer   )r"   r    r   r   r#   get_sheet_by_name   s    
z OpenpyxlReader.get_sheet_by_nameint)rp   c                 C  s   |  | | jj| S r9   )Zraise_if_bad_sheet_by_indexr   r+   )r"   rp   r   r   r#   get_sheet_by_index$  s    
z!OpenpyxlReader.get_sheet_by_indexboolr   )convert_floatr8   c                 C  sf   ddl m}m} |jd krdS |j|kr.tjS |j|kr`|rVt|j}||jkr`|S n
t|jS |jS )Nr   )
TYPE_ERRORTYPE_NUMERIC )	Zopenpyxl.cell.cellr   r   rw   Z	data_typenpnanr   float)r"   rs   r   r   r   rv   r   r   r#   _convert_cell(  s    





zOpenpyxlReader._convert_cellzlist[list[Scalar]]c                   s   j jr|  g }d}t|jD ]H\}} fdd|D }|rX|d dkrX|  q>|r`|}|| q"|d |d  }t|dkrtdd |D t	d	d |D k rdgfd
d|D }|S )Nc                   s   g | ]} | qS r   )r   )r   rs   )r   r"   r   r#   r   F  s     z1OpenpyxlReader.get_sheet_data.<locals>.<listcomp>r   rm   r   c                 s  s   | ]}t |V  qd S r9   lenr   Zdata_rowr   r   r#   	<genexpr>S  s     z0OpenpyxlReader.get_sheet_data.<locals>.<genexpr>c                 s  s   | ]}t |V  qd S r9   r   r   r   r   r#   r   T  s     c                   s    g | ]}|t |    qS r   r   r   )
empty_cell	max_widthr   r#   r   V  s   )
r   r   Zreset_dimensions	enumerateZrowspopappendr   maxmin)r"   r   r   dataZlast_row_with_dataZ
row_numberrn   Zconverted_rowr   )r   r   r   r"   r#   get_sheet_data>  s(    
zOpenpyxlReader.get_sheet_data)N)r   r   r   r&   propertyr   r   r   r   r   r   r   r   r   r   r0   r#   r     s    
r   )
__future__r   r4   typingr   r   Znumpyr   Zpandas._typingr   r   r   r   Zpandas.compat._optionalr	   Zpandas.io.excel._baser
   r   Zpandas.io.excel._utilr   r   Z!openpyxl.descriptors.serialisabler   r   r   r   r   r   r#   <module>   s      _