a
    w=ic                     @   sV  d dl mZ d dlZd dlZd dlZd dlmZ d dlmZ d dl	m
Z
 d dlmZmZmZmZmZ d dlmZmZmZ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 d	l!m"Z"m#Z# d dl$Z$e$%d
Z&dZ'z0zd dl(Z)W n e*y   d dl)Z)Y n0 dZ'W n e*y   Y n0 G dd deZ+G dd deZ,dZ-e.e-Z/dZ0dZ1e.e1Z2dZ3e.e3Z4dZ5dZ6e2e4 de6  Z7dZ8dZ9e.e9Z:dZ;dd Z<G dd deZ=G dd  d ed!Z>G d"d# d#ed$Z?G d%d& d&ed'Z@G d(d) d)eZAd*d+ ZBd,d- ZCd.d/ ZDd0d1 ZEd2d3 ZFd;d4d5ZGd6d7 ZHd<d8d9ZIeJd:krReKeI  dS )=    )BytesION)OrderedDict)sstruct)calcIntBounds)Tagbytechrbyteord	bytesjoinpad)TTFont
TTLibErrorgetTableModulegetTableClassgetSearchRange)	
SFNTReader
SFNTWriterDirectoryEntryWOFFFlavorDatasfntDirectoryFormatsfntDirectorySizeSFNTDirectoryEntrysfntDirectoryEntrySizecalcChecksum)	ttProgram_g_l_y_fzfontTools.ttLib.woff2FTc                   @   sL   e Zd ZdZdddZdd Zdd	 ZdddZdd Zdd Z	dd Z
d
S )WOFF2Readerwoff2r   c                 C   s\  t std td|| _t| jd}|dkr<td| jd t	| _
| jt}t|tkrntdtt||  t | _d}t| jD ]<}| 
 }|| j t|j}	|| j|	< ||_||j7 }q|}
| j| j}t|}t||
krtd|
t|f t|| _| jdd	 | j| j kr@td
t| | _t ddd| _!d S )NzfThe WOFF2 decoder requires the Brotli Python extension, available at: https://github.com/google/brotliNo module named brotli   s   wOF2z Not a WOFF2 font (bad signature)r   z"Not a WOFF2 font (not enough data)zAunexpected size for decompressed font data: expected %d, found %d   z4reported 'length' doesn't match the actual file sizeFZrecalcBBoxesZrecalcTimestamp)"
haveBrotlilogerrorImportErrorfiler   readr   seekWOFF2DirectoryEntryr   woff2DirectorySizelenr   unpackwoff2DirectoryFormatr   tablesrange	numTablesfromFiletagoffsetlengthtotalCompressedSizebrotli
decompressr   transformBuffertellWOFF2FlavorData
flavorDatar   ttFont)selfr&   ZcheckChecksumsZ
fontNumber	signaturedatar3   ientryr2   ZtotalUncompressedSizeZcompressedDataZdecompressedData rB   f/home/droni/.local/share/virtualenvs/DPS-5Je3_V2c/lib/python3.9/site-packages/fontTools/ttLib/woff2.py__init__#   sN    





zWOFF2Reader.__init__c                 C   s@   | j t| }t|ds:|jr,| ||_n|| j|_|jS )z9Fetch the raw table data. Reconstruct transformed tables.r?   )r.   r   hasattrtransformedreconstructTabler?   loadDatar8   )r=   r2   rA   rB   rB   rC   __getitem__S   s    
zWOFF2Reader.__getitem__c                 C   sz   | j t| }|| j}|dkrDt| dr2| jnd}| ||}n2|dkrV|  }n |dkrj| |}nt	d| |S )z4Reconstruct table named 'tag' from transformed data.glyfpaddingNlocahmtxz#transform for table '%s' is unknown)
r.   r   rH   r8   rE   rK   _reconstructGlyf_reconstructLoca_reconstructHmtxr   )r=   r2   rA   rawDatarK   r?   rB   rB   rC   rG   ]   s    
zWOFF2Reader.reconstructTableNc                 C   sD   t  | jd< t  }| jd< ||| j |r4||_|| j}|S )z Return recostructed glyf table data, and set the corresponding loca's
		locations. Optionally pad glyph offsets to the specified number of bytes.
		rL   rJ   )WOFF2LocaTabler<   WOFF2GlyfTablereconstructrK   compile)r=   r?   rK   	glyfTablerB   rB   rC   rN   m   s    zWOFF2Reader._reconstructGlyfc                 C   sf   d| j vr| d| jd _| j d }|| j }t|| jd jkrbtd| jd jt|f |S )z' Return reconstructed loca table data. rL   rJ   zMreconstructed 'loca' table doesn't match original size: expected %d, found %d)r<   rG   r.   r?   rU   r+   
origLengthr   )r=   Z	locaTabler?   rB   rB   rC   rO   y   s    

zWOFF2Reader._reconstructLocac                 C   sX   d| j jv rd}nd}|D ]}| | qt  }| jd< ||| j || j}|S )z' Return reconstructed hmtx table data. rJ   )maxphhearJ   )rX   headrY   rL   rJ   rM   )r;   transformedTables_decompileTableWOFF2HmtxTabler<   rT   rU   )r=   r?   ZtableDependenciesr2   	hmtxTablerB   rB   rC   rP      s    zWOFF2Reader._reconstructHmtxc                 C   sL   | | }| j |r| j | S t|}||}|| j j|< ||| j  dS )z5Decompile table data and store it inside self.ttFont.N)r<   isLoadedr   r.   	decompiler=   r2   r?   Z
tableClasstablerB   rB   rC   r\      s    
zWOFF2Reader._decompileTable)r   r   )N)__name__
__module____qualname__flavorrD   rI   rG   rN   rO   rP   r\   rB   rB   rB   rC   r      s   
0

r   c                   @   s   e Zd ZdZd)ddZdd Zdd	 Zd*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dd  Zd!d" Zd#d$ Zd%d& Zd'd( ZdS )+WOFF2Writerr      Nc                 C   s|   t std td|| _|| _t|| _t|d| _	t
| _t| _t| _td| _d| _t | _t | _tddd| _d S )NzfThe WOFF2 encoder requires the Brotli Python extension, available at: https://github.com/google/brotlir   )r?   ZwOF2r   Fr!   )r"   r#   r$   r%   r&   r0   r   sfntVersionr:   r;   r-   directoryFormatr*   directorySizer)   r   r>   nextTableOffsetr   r8   r   r.   r   r<   )r=   r&   r0   ri   rf   r;   rB   rB   rC   rD      s"    

zWOFF2Writer.__init__c                 C   sb   || j v rtd| |dkr0|  jd8  _dS |  }t||_t|j|_||_|| j |< dS )z4Associate new entry named 'tag' with raw table data.zcannot rewrite '%s' tableZDSIG   N)	r.   r   r0   r   r   r2   getKnownTagIndexflagsr?   )r=   r2   r?   rA   rB   rB   rC   __setitem__   s    

zWOFF2Writer.__setitem__c                 C   s  t | j| jkr(td| jt | jf | jdv r8d}n| jdkrHd}ntd|rvd| jjv rvd| jv rv| jdd	 |   t	t
| j | _|  | _|  }tj|tjd
}t || _|  | _|  \| _| _d| _|  }| jd | jt|| dd |   dS )zN All tags must have been specified. Now write the table data and directory.
		z-wrong number of tables; expected %d, found %d)rh   trueTOTTOFz1Not a TrueType or OpenType font (bad sfntVersion)rJ   r   )rK   moder   sizeN)r+   r.   r0   r   ri   r;   r[   _normaliseGlyfAndLoca_setHeadTransformFlagr   sorteditems#_calcSFNTChecksumsLengthsAndOffsetsZtotalSfntSize_transformTablesr6   compressZ	MODE_FONTr5   _calcTotalSizer4   _getVersionmajorVersionminorVersionreserved_packTableDirectoryr&   r(   writer
   _writeFlavorData)r=   Z
isTrueTypefontDataZcompressedFont	directoryrB   rB   rC   close   s6    

	



zWOFF2Writer.closer   c                 C   sF   | j dkrdS dD ]}| | q|| jd _dD ]}| | q2dS )z Recompile glyf and loca tables, aligning glyph offsets to multiples of
		'padding' size. Update the head table's 'indexToLocFormat' accordingly while
		compiling loca.
		rr   NrX   rZ   rL   rJ   rJ   rJ   rL   )ri   r\   r<   rK   _compileTable)r=   rK   r2   rB   rB   rC   rw     s    
z!WOFF2Writer._normaliseGlyfAndLocac                 C   s,   |  d | jd  jdO  _| d dS )z Set bit 11 of 'head' table flags to indicate that the font has undergone
		a lossless modifying transform. Re-compile head table data.rZ   i   N)r\   r<   ro   r   r=   rB   rB   rC   rx     s    
z!WOFF2Writer._setHeadTransformFlagc                 C   s   t |}|| jvrtd| | j|r.dS | j| j}|dkrHt}n$|dkrVt}n|dkrdt}nt	|}||}|| jj|< |
|| j dS )zB Fetch table data, decompile it, and store it inside self.ttFont. zmissing required table: %sNrL   rJ   rM   )r   r.   r   r<   r_   r?   rR   rS   r]   r   r`   ra   rB   rB   rC   r\     s     
zWOFF2Writer._decompileTablec                 C   s   | j | | j | j| _dS )z5 Compile table and store it in its 'data' attribute. N)r<   rU   r.   r?   r=   r2   rB   rB   rC   r   0  s    zWOFF2Writer._compileTablec                 C   s   t tt| j  }| j D ]f\}}|j}||_t||_|dkrft|dd d |dd  |_	n
t||_	||jd d@ 7 }q|S )z Compute the 'original' SFNT checksums, lengths and offsets for checksum
		adjustment calculation. Return the total size of the uncompressed font.
		rZ   N   s             )
r   r   r+   r.   rz   r?   
origOffsetrW   r   checkSum)r=   r3   r2   rA   r?   rB   rB   rC   r{   4  s    
$
z/WOFF2Writer._calcSFNTChecksumsLengthsAndOffsetsc                 C   s   | j j}| j D ]f\}}d}||v r>| |}|dur>d|_|du rR|j}d|_| j|_|	| j
| |  j|j7  _q|   | j
 }|S )zReturn transformed font data.NTF)r;   r[   r.   rz   transformTablerF   r?   rl   r3   ZsaveDatar8   r4   writeMasterChecksumgetvalue)r=   r[   r2   rA   r?   r   rB   rB   rC   r|   D  s     

zWOFF2Writer._transformTablesc                 C   s   |dkrd}n|dkrBdD ]}|  | q| jd }|| j}nN|dkrd| jvrXdS dD ]}|  | q\| jd }|| j}ntd| |S )	zReturn transformed table data, or None if some pre-conditions aren't
		met -- in which case, the non-transformed table data will be used.
		rL       rJ   r   rM   N)rX   rZ   rY   rL   rJ   rM   z#Transform for table '%s' is unknown)r\   r<   	transformr.   r   )r=   r2   r?   rV   r^   rB   rB   rC   r   X  s     


zWOFF2Writer.transformTablec                 C   s   t | j }g }tt|D ]}|| j||  j qt| jd\| _	| _
| _tt| }t| j }|D ]:\}}t }|j|_|j|_|j|_|j|_||  }qptt| jt  }	|	t|ksJ |t| t|d@ }
d|
 d@ }|S )zCalculate checkSumAdjustment.   l    l   /ac )listr.   keysr/   r+   appendr   r   r0   ZsearchRangeZentrySelectorZ
rangeShiftr   packr   ry   rz   r   r2   r   r3   rW   r4   toStringr   r   r   sum)r=   tagsZ	checksumsr@   r   r.   r2   rA   Z	sfntEntryZdirectory_endchecksumchecksumadjustmentrB   rB   rC   _calcMasterChecksumn  s(    zWOFF2Writer._calcMasterChecksumc                 C   s8   |   }| j| jd jd  | jtd| dS )z0Write checkSumAdjustment to the transformBuffer.rZ   r   z>LN)r   r8   r(   r.   r3   r   structr   )r=   r   rB   rB   rC   r     s    zWOFF2Writer.writeMasterChecksumc                 C   sJ   | j }| j D ]}|t| 7 }q|| j7 }|d d@ }| |}|S )zLCalculate total size of WOFF2 font, including any meta- and/or private data.r   r   )rk   r.   valuesr+   r   r5   _calcFlavorDataOffsetsAndSize)r=   r3   rA   rB   rB   rC   r~     s    

zWOFF2Writer._calcTotalSizec                 C   s   |}| j }|jrNt|j| _|| _tj|jtjd| _t| j| _	|| j	7 }nd | _ | _	| _d| _|j
r|d d@ }|| _t|j
| _|| j7 }nd | _| _|S )z@Calculate offsets and lengths for any meta- and/or private data.rs   r   r   r   r   )r;   metaDatar+   ZmetaOrigLength
metaOffsetr6   r}   Z	MODE_TEXTcompressedMetaDataZ
metaLengthprivData
privOffsetZ
privLength)r=   startr3   r?   rB   rB   rC   r     s&    z)WOFF2Writer._calcFlavorDataOffsetsAndSizec                 C   sT   | j }|jdur&|jdur&|j|jfS d| jv rLtd| jd jdd S dS dS )z;Return the WOFF2 font's (majorVersion, minorVersion) tuple.NrZ   z>HHr   r   )r   r   )r;   r   r   r.   r   r,   r?   r=   r?   rB   rB   rC   r     s    
zWOFF2Writer._getVersionc                 C   s.   t | j| }| j D ]}||  }q|S )z"Return WOFF2 table directory data.)r   r   rj   r.   r   r   )r=   r   rA   rB   rB   rC   r     s    zWOFF2Writer._packTableDirectoryc                 C   s   | j }| jj}|r"|r"t|dd}|rT| j| j | j | jksHJ | j| |r| j| j	 | j | j	kszJ | j| dS )z<Write metadata and/or private data using appropiate padding.r   ru   N)
r   r;   r   r
   r&   r(   r   r9   r   r   )r=   r   r   rB   rB   rC   r     s    zWOFF2Writer._writeFlavorDatac                 C   s   dS )NTrB   r   rB   rB   rC   reordersTables  s    zWOFF2Writer.reordersTables)rh   NN)r   )rc   rd   re   rf   rD   rp   r   rw   rx   r\   r   r{   r|   r   r   r   r~   r   r   r   r   r   rB   rB   rB   rC   rg      s*     
2

rg   a  
		> # big endian
		signature:           4s   # "wOF2"
		sfntVersion:         4s
		length:              L    # total woff2 file size
		numTables:           H    # number of tables
		reserved:            H    # set to 0
		totalSfntSize:       L    # uncompressed size
		totalCompressedSize: L    # compressed size
		majorVersion:        H    # major version of WOFF file
		minorVersion:        H    # minor version of WOFF file
		metaOffset:          L    # offset to metadata block
		metaLength:          L    # length of compressed metadata
		metaOrigLength:      L    # length of uncompressed metadata
		privOffset:          L    # offset to private data block
		privLength:          L    # length of private data block
)?ZcmaprZ   rY   rM   rX   namezOS/2postzcvt ZfpgmrJ   rL   prepzCFF ZVORGZEBDTZEBLCZgaspZhdmxkernZLTSHZPCLTZVDMXZvheaZvmtxZBASEZGDEFZGPOSZGSUBZEBSCZJSTFZMATHZCBDTZCBLCZCOLRZCPALzSVG ZsbixZacntZavarZbdatZblocZbslnZcvarZfdscZfeatZfmtxZfvarZgvarZhstyZjustZlcarZmortZmorxZopbdpropZtrakZZapfZSilfZGlatZGlocZFeatZSillz5
		> # big endian
		flags: B  # table type and flags
z5
		> # big endian
		tag: 4s  # 4-byte tag (optional)
?      r    r   aN  
		> # big endian
		version:                  L  # = 0x00000000
		numGlyphs:                H  # Number of glyphs
		indexFormat:              H  # Offset format for loca table
		nContourStreamSize:       L  # Size of nContour stream
		nPointsStreamSize:        L  # Size of nPoints stream
		flagStreamSize:           L  # Size of flag stream
		glyphStreamSize:          L  # Size of glyph stream
		compositeStreamSize:      L  # Size of composite stream
		bboxStreamSize:           L  # Comnined size of bboxBitmap and bboxStream
		instructionStreamSize:    L  # Size of instruction stream
zF
		>	# big endian
		xMin:				h
		yMin:				h
		xMax:				h
		yMax:				h
c                 C   s*   t ttD ]}| t| kr|  S qtS )zEReturn index of 'tag' in woff2KnownTags list. Return 63 if not found.)r/   r+   woff2KnownTagswoff2UnknownTagIndex)r2   r@   rB   rB   rC   rn   %  s    
rn   c                   @   sX   e Zd Zdd Zdd Zdd Zedd Zejd	d Zed
d Z	e	jdd Z	dS )r)   c                 C   s>   |  }|t}| |}t|t| }|||  d S N)r9   r'   woff2DirectoryEntryMaxSize
fromStringr+   r(   )r=   r&   posr?   leftconsumedrB   rB   rC   r1   /  s
    

zWOFF2DirectoryEntry.fromFilec                 C   s   t |dk rtdtt|| \}}| jd@ dkr\t |tk rHtdtt|| \}}nt| jd@  | _	t
| j	| _	t|\| _}| j| _| jrt|\| _}| j	dkr| jdkrtd|S )Nrm   z)can't read table 'flags': not enough datar   z'can't read table 'tag': not enough datarL   r   z1the transformLength of the 'loca' table must be 0)r+   r   r   unpack2woff2FlagsFormatro   woff2UnknownTagSizewoff2UnknownTagFormatr   r2   r   unpackBase128rW   r4   rF   )r=   r?   dummyrB   rB   rC   r   6  s$    zWOFF2DirectoryEntry.fromStringc                 C   sT   t | j}| jd@ dkr.|td| j 7 }|t| j7 }| jrP|t| j	7 }|S )Nr   z>4s)
r   ro   r   r   r2   tobytespackBase128rW   rF   r4   r   rB   rB   rC   r   M  s    
zWOFF2DirectoryEntry.toStringc                 C   s
   | j d? S )z~Return bits 6-7 of table entry's flags, which indicate the preprocessing
		transformation version number (between 0 and 3).
		   ro   r   rB   rB   rC   transformVersionV  s    z$WOFF2DirectoryEntry.transformVersionc                 C   s0   d|  krdksn J |  j |d> O  _ d S )Nr   r   r   r   )r=   valuerB   rB   rC   r   ]  s    c                 C   s"   | j dv r| jdkS | jdkS dS )zCReturn True if the table has any transformation, else return False.>   rL   rJ   r   r   N)r2   r   r   rB   rB   rC   rF   b  s    

zWOFF2DirectoryEntry.transformedc                 C   s(   | j dv r|sdnd| _n
t|| _d S )N>   rL   rJ   r   r   )r2   r   int)r=   ZbooleanValuerB   rB   rC   rF   n  s    
N)
rc   rd   re   r1   r   r   propertyr   setterrF   rB   rB   rB   rC   r)   -  s   	


r)   c                       s*   e Zd ZdZdddZ fddZ  ZS )rR   zSame as parent class. The only difference is that it attempts to preserve
	the 'indexFormat' as encoded in the WOFF2 glyf table.
	Nc                 C   s   t |pd| _d S )NrL   r   tableTagr   rB   rB   rC   rD   ~  s    zWOFF2LocaTable.__init__c                    s   zt | j}W n  ty.   | g  d}Y n0 d|v rt|d dr|d j}|dkr|dkrhtdtdd | jD stdtd	}t	t
| jD ]}|| j| d
  qntd| j}tjdkr|  | }ntt| |}|S )Nr   rJ   indexFormati   z,indexFormat is 0 but local offsets > 0x20000c                 s   s   | ]}|d  dkV  qdS )r    r   NrB   ).0lrB   rB   rC   	<genexpr>  r   z)WOFF2LocaTable.compile.<locals>.<genexpr>z5indexFormat is 0 but local offsets not multiples of 2Hr    Ibig)max	locationsAttributeErrorsetrE   r   r   allarrayr/   r+   r   sys	byteorderbyteswapr   superrR   rU   )r=   r<   Zmax_locationr   r   r@   r?   	__class__rB   rC   rU     s(    




zWOFF2LocaTable.compile)N)rc   rd   re   __doc__rD   rU   __classcell__rB   rB   r   rC   rR   y  s   
rR   rL   c                   @   s   e Zd ZdZdZd"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dd Zdd Zdd Zdd Zd d! ZdS )#rS   z1Decoder/Encoder for WOFF2 'glyf' table transform.)nContourStreamnPointsStream
flagStreamglyphStreamcompositeStream
bboxStreaminstructionStreamNc                 C   s   t |pd| _d S )NrJ   r   r   rB   rB   rC   rD     s    zWOFF2GlyfTable.__init__c                 C   s  t |}|tk rtdtt|| \}}t}| jD ]:}t| |d }t| ||d|  ||d }||7 }q4||krtd||f | j	d d? d> }| j
d| }	td|	| _| j
|d | _
td	| j| _tjd
kr| j  t | j| j	ksJ d|v r| j|d _z| | _W n   d| _Y n0 | jdu rjdg| _| jdd td| j	D  n*t | j| j	krtdt | j| j	f i  }
| _t| jD ]\}}| |}||
|< qdS )z$ Decompile transformed 'glyf' data. znot enough 'glyf' dataSizeNzJincorrect size of transformed 'glyf' table: expected %d, received %d bytes   r   r    Bhr   rZ   z.notdefc                 S   s   g | ]}d | qS )z	glyph%.5drB   r   r@   rB   rB   rC   
<listcomp>  r   z.WOFF2GlyfTable.reconstruct.<locals>.<listcomp>rm   z2incorrect glyphOrder: expected %d glyphs, found %d)r+   woff2GlyfTableFormatSizer   r   r   woff2GlyfTableFormat
subStreamsgetattrsetattr	numGlyphsr   r   
bboxBitmapr   r   r   r   r   indexToLocFormatgetGlyphOrder
glyphOrderextendr/   glyphs	enumerate_decodeGlyph)r=   r?   r<   ZinputDataSizer   r3   streamrv   bboxBitmapSizer   r   glyphID	glyphNameglyphrB   rB   rC   rT     sT    


 

zWOFF2GlyfTable.reconstructc                    s  t  j _t  j jks J d|v r4 j|d _|d j _ jD ]}t |d qF jd d? d> }tddg|  _	t
 jD ]} | q j	  j  _ jD ]}t |d	 t t | qd _tt }|t fd
d jD 7 }|S )z  Return transformed 'glyf' data rX   rZ   r   r   r   r    r   r   r   c                    s   g | ]}t  |qS rB   )r   )r   sr   rB   rC   r     r   z,WOFF2GlyfTable.transform.<locals>.<listcomp>)r+   r   r   r   r   r   r   r   r   r   r/   _encodeGlyphr   r   r   versionr   r   r   r	   )r=   r<   r   r   r  r?   rB   r   rC   r     s$    

zWOFF2GlyfTable.transformc                 C   sT   t d }| j| |_|jdkr&|S | r:| | n
| | | || |S )NrJ   r   )r   ZGlyphr   numberOfContoursisComposite_decodeComponents_decodeCoordinates_decodeBBox)r=   r  r  rB   rB   rC   r     s    

zWOFF2GlyfTable._decodeGlyphc                 C   sd   | j }g |_d}d}|rLtd }||| \}}}||B }|j| q|| _ |r`| | d S )Nrm   r   rJ   )r   
componentsr   ZGlyphComponentr`   r   _decodeInstructions)r=   r  r?   morehaveInstructions	componentZ	haveInstrrB   rB   rC   r	    s    z WOFF2GlyfTable._decodeComponentsc                 C   s`   | j }g }d}t|jD ]"}t|\}}||7 }|| q||_|| _ | | | | d S Nr   )r   r/   r  unpack255UShortr   endPtsOfContours_decodeTripletsr  )r=   r  r?   r  endPointr@   ptsOfContourrB   rB   rC   r
  	  s    
z!WOFF2GlyfTable._decodeCoordinatesc                 C   sN   | j }| j}t|\}}t |_|j|d |  || _ ||d  | _d S r   )r   r   r  r   ZProgramprogramZfromBytecode)r=   r  r   r   ZinstructionLengthrB   rB   rC   r    s    
z"WOFF2GlyfTable._decodeInstructionsc                 C   s`   t | j|d?  d|d@ ? @ }| r6|s6td| |rRtt| j|\}| _n
||  d S )Nr         z%no bbox values for composite glyph %d)	boolr   r  r   r   r   
bboxFormatr   ZrecalcBounds)r=   r  r  ZhaveBBoxr   rB   rB   rC   r    s    zWOFF2GlyfTable._decodeBBoxc                 C   s
  dd }|j d d }|}|t| jkr0td| jd | }| j|d  | _td|}td| j}t|}||ks|J d}	d}
tdj||_	td|_
d}t|D ]@}|| }t|d	?  }|d
M }|dk rd}n |dk rd}n|dk rd}nd}|| |ksJ |dk r>d}|||d@ d	> ||  }n||dk rn|||d d@ d	> ||  }d}nL|dk r|d }|| }||d|d@  |d?  }||d? d|d@ d>  |d@  }n|dk r"|d }||d|d d>  ||  }||d? d|d d? d>  ||d   }n|dk rv||d  }|||| d> |d?  }||d? |d@ d> ||d   }nD|||| d> ||d   }||d? ||d  d> ||d   }||7 }|	|7 }	|
|7 }
|	|
f|j	|< |j
t| q|}| j|d  | _d S )Nc                 S   s*   d|kr|dk sJ d| d@ r$|S | S )Nr   i   zinteger overflowrm   rB   )flagZbasevalrB   rB   rC   withSign*  s    z0WOFF2GlyfTable._decodeTriplets.<locals>.withSignr   rm   znot enough 'flagStream' datar   r   rJ   r     T   x   r    |   r   r   
         0   r      r   )r  r+   r   r   r   r   r   ZGlyphCoordinatesZzeroscoordinatesro   r/   r  r   r   )r=   r  r  ZnPointsZflagSizeZ	flagsDataro   tripletsZ	nTripletsxyZtripletIndexr@   r  onCurveZnBytesZdxZdyZb0b1b2ZbytesConsumedrB   rB   rC   r  (  s    


$

zWOFF2GlyfTable._decodeTripletsc                 C   sf   |  |}| | }|  jtd|j7  _|jdkr8d S | rL| | n
| | | || d S )Nz>hr   )	ZgetGlyphNamer   r   r   r  r  _encodeComponents_encodeCoordinates_encodeBBox)r=   r  r  r  rB   rB   rC   r  n  s    


zWOFF2GlyfTable._encodeGlyphc                 C   st   t |jd }d}d}tt |jD ]<}||kr>t|d}d}|j| }|  j|||| 7  _q$|rp| | d S )Nrm   r   r  )r+   r  r/   rE   r   rU   _encodeInstructions)r=   r  Zlastcomponentr  r  r@   r  rB   rB   rC   r.  z  s    

z WOFF2GlyfTable._encodeComponentsc                 C   sF   d}|j D ]"}|| }|  jt|7  _|}q
| | | | d S r  )r  r   pack255UShort_encodeTripletsr1  )r=   r  ZlastEndPointr  r  rB   rB   rC   r/    s    

z!WOFF2GlyfTable._encodeCoordinatesc                 C   s2   |j  }|  jtt|7  _|  j|7  _d S r   )r  ZgetBytecoder   r2  r+   r   )r=   r  ZinstructionsrB   rB   rC   r1    s    
z"WOFF2GlyfTable._encodeInstructionsc                 C   s|   |j dksJ d| sD|j|j|j|jf}t|j}||krDd S | j|d?   d|d@ ? O  < |  j	t
t|7  _	d S )Nr   zempty glyph has no bboxr   r  r  )r  r  xMinZyMinZxMaxZyMaxr   r'  r   r   r   r   r  )r=   r  r  ZcurrentBBoxZcalculatedBBoxrB   rB   rC   r0    s    
zWOFF2GlyfTable._encodeBBoxc                 C   s  t |jt |jksJ |j }|  td}td}tt |D ]b}|j| tj@ }|| \}}t	|}	t	|}
|rdnd}|dk rdnd}|dk rdnd}|d|  }|dkr|
dk r|
||
d@ d?  |  |
|
d	@  qJ|dkr0|	dk r0|
|d
 |	d@ d?  |  |
|	d	@  qJ|	dk r|
dk r|
|d |	d d@  |
d d@ d?  |  |
|	d d@ d> |
d d@ B  qJ|	dk r|
dk r|
|d d|	d d@ d?   |
d d@ d?  |  |
|	d d	@  |
|
d d	@  qJ|	dk rd|
dk rd|
|d |  |
|	d?  |
|	d@ d> |
d? B  |
|
d	@  qJ|
|d |  |
|	d?  |
|	d	@  |
|
d?  |
|
d	@  qJ|  j| 7  _|  j| 7  _d S )Nr   r   r  rm   r    i   i   r     r"  A   r$  r%  r&  r   i  r  r   i   r   r   i   r   r!  )r+   r'  ro   copyZabsoluteToRelativer   r/   r   ZflagOnCurveabsr   r   r   r   )r=   r  r'  ro   r(  r@   r+  r)  r*  ZabsXZabsYZ
onCurveBitZxSignBitZySignBitZ
xySignBitsrB   rB   rC   r3    sN    


.$6zWOFF2GlyfTable._encodeTriplets)N)rc   rd   re   r   r   rD   rT   r   r   r	  r
  r  r  r  r  r.  r/  r1  r0  r3  rB   rB   rB   rC   rS     s"   
1		F	rS   rJ   c                   @   s&   e Zd ZdddZdd Zdd ZdS )	r]   Nc                 C   s   t |pd| _d S NrM   r   r   rB   rB   rC   rD     s    zWOFF2HmtxTable.__init__c                 C   s  t d|d d \}|dd  }|d@ dkr<td| j |d@ dk}|d@ dk}|rj|rjtd| j |d }|d	 }|j}t|}	tt|j|	}
t|d|
 ksJ t		d
|d d|
  }t
jdkr|  |d|
 d  }|r@t|d|
 ksJ t		d|d d|
  }t
jdkr.|  |d|
 d  }nLt		d}t|D ]8\}}||
krj q|| }t|dd}|| qR|	|
 }|rt|d| ksJ t		d|d d|  }t
jdkr|  |d| d  }nJt		d}t|D ]6\}}||
k rq || }t|dd}|| q |rLtd| j i | _t|
D ].}|| }|| ||  }}||f| j|< qZ|d }t|D ]$}|||
  }||| f| j|< qd S )N>Brm      r   z#Bits 2-7 of '%s' flags are reservedr    z?either bits 0 or 1 (or both) must set in transformed '%s' flagsrJ   rY   r   r   r   r4  ztoo much '%s' table datar   )r   r,   r   r   r   r+   minr   numberOfHMetricsr   r   r   r   r   r   r   metricsr/   )r=   r?   r<   ro   hasLsbArrayhasLeftSideBearingArrayrV   ZheaderTabler   r   r=  advanceWidthArraylsbArrayr@   r  r  r4  ZnumberOfSideBearingsleftSideBearingArrayZadvanceWidthlsbZlastAdvancerB   rB   rC   rT     sx    




zWOFF2HmtxTable.reconstructc                    s  |   |d }|d }|jd}tD ]6} | }j| d }|t|| ddkr*d} qbq*d}tt D ]6} | }j| d }|t|| ddkrtd} qqt|r|rd S d}	|s|	dO }	|s|	dO }	td	|	}
td
fddt	 D }t
jdkr|  |
| 7 }
|rftdfddt	 D }t
jdkrZ|  |
| 7 }
|rtd fddtt D }t
jdkr|  |
| 7 }
|
S )NrJ   rY   Frm   r4  r   Tr    r:  r   c                    s&   g | ]\}}| k rj | d  qS )r   r>  r   r@   r  r=  r=   rB   rC   r   G  s   z,WOFF2HmtxTable.transform.<locals>.<listcomp>r   r   c                    s&   g | ]\}}| k rj | d  qS rm   rE  rF  rG  rB   rC   r   T  s   c                    s   g | ]}j  |  d  qS rH  rE  r   )r   r=   rB   rC   r   a  s   )r   r=  r/   r>  r   r+   r   r   r   r   r   r   r   r   )r=   r<   rJ   rY   r?  r@   r  rD  r@  ro   r?   rA  rB  rC  rB   )r   r=  r=   rC   r     sp    zWOFF2HmtxTable.transform)N)rc   rd   re   rD   rT   r   rB   rB   rB   rC   r]     s   
Jr]   rM   c                       s*   e Zd ZdZd fdd	Zdd Z  ZS )r:   r   Nc                    s   t std|dur4|dur$td|dur4td|durdd|v rLd|vs\d|v rdd|vrdtdtt| j|d |rd	d
 |j D }n<|r|j	| _	|j
| _	|j| _|j| _|du rt|dr|j}|du rt}t|| _dS )a  Data class that holds the WOFF2 header major/minor version, any
		metadata or private data (as bytes strings), and the set of
		table tags that have transformations applied (if reader is not None),
		or will have once the WOFF2 font is compiled.

		Args:
			reader: an SFNTReader (or subclass) object to read flavor data from.
			data: another WOFFFlavorData object to initialise data from.
			transformedTables: set of strings containing table tags to be transformed.

		Raises:
			ImportError if the brotli module is not installed.

		NOTE: The 'reader' argument, on the one hand, and the 'data' and
		'transformedTables' arguments, on the other hand, are mutually exclusive.
		r   Nz4'reader' and 'data' arguments are mutually exclusivezA'reader' and 'transformedTables' arguments are mutually exclusiverJ   rL   z7'glyf' and 'loca' must be transformed (or not) together)readerc                 S   s   g | ]\}}|j r|qS rB   )rF   )r   r2   rA   rB   rB   rC   r     s   z,WOFF2FlavorData.__init__.<locals>.<listcomp>r[   )r"   r%   	TypeError
ValueErrorr   r:   rD   r.   rz   r   r   r   r   rE   r[   woff2TransformedTableTagsr   )r=   rI  r?   r[   r   rB   rC   rD   q  sL    zWOFF2FlavorData.__init__c                 C   s
   t |S r   )r6   r7   )r=   rQ   rB   rB   rC   _decompress  s    zWOFF2FlavorData._decompress)NNN)rc   rd   re   ZFlavorrD   rM  r   rB   rB   r   rC   r:   m  s   9r:   c                 C   s   t | dkrtdd}t| d dkr0tdttD ]h}t | dkrPtdt| d }| dd } |d@ rxtd|d	> |d
@ B }|d@ dkr8|| f  S q8tddS )a   Read one to five bytes from UIntBase128-encoded input string, and return
	a tuple containing the decoded integer plus any leftover data.

	>>> unpackBase128(b'\x3f\x00\x00') == (63, b"\x00\x00")
	True
	>>> unpackBase128(b'\x8f\xff\xff\xff\x7f')[0] == 4294967295
	True
	>>> unpackBase128(b'\x80\x80\x3f')  # doctest: +IGNORE_EXCEPTION_DETAIL
	Traceback (most recent call last):
	  File "<stdin>", line 1, in ?
	TTLibError: UIntBase128 value must not start with leading zeros
	>>> unpackBase128(b'\x8f\xff\xff\xff\xff\x7f')[0]  # doctest: +IGNORE_EXCEPTION_DETAIL
	Traceback (most recent call last):
	  File "<stdin>", line 1, in ?
	TTLibError: UIntBase128-encoded sequence is longer than 5 bytes
	>>> unpackBase128(b'\x90\x80\x80\x80\x00')[0]  # doctest: +IGNORE_EXCEPTION_DETAIL
	Traceback (most recent call last):
	  File "<stdin>", line 1, in ?
	TTLibError: UIntBase128 value exceeds 2**32-1
	r   z%not enough data to unpack UIntBase128r  z3UIntBase128 value must not start with leading zerosrm   Nl      | z!UIntBase128 value exceeds 2**32-1r  r  z3UIntBase128-encoded sequence is longer than 5 bytes)r+   r   r   r/   woff2Base128MaxSize)r?   resultr@   coderB   rB   rC   r     s     r   c                 C   s.   | dksJ d}| dkr*|d7 }| dL } q|S )z Return the length in bytes of a UIntBase128-encoded sequence with value n.

	>>> base128Size(0)
	1
	>>> base128Size(24567)
	3
	>>> base128Size(2**32-1)
	5
	r   rm   r  r  rB   )nrv   rB   rB   rC   base128Size  s    

rR  c                 C   sr   | dk s| dkrt dd}t| }t|D ]@}| d|| d  ? d@ }||d k r\|dO }|td	|7 }q,|S )
a	   Encode unsigned integer in range 0 to 2**32-1 (inclusive) to a string of
	bytes using UIntBase128 variable-length encoding. Produce the shortest possible
	encoding.

	>>> packBase128(63) == b"\x3f"
	True
	>>> packBase128(2**32-1) == b'\x8f\xff\xff\xff\x7f'
	True
	r   l        z3UIntBase128 format requires 0 <= integer <= 2**32-1r   r  rm   r  r  r   )r   rR  r/   r   r   )rQ  r?   rv   r@   brB   rB   rC   r     s    
r   c                 C   s   t | dd }| dd } |dkr\t| dk r8tdtd| dd \}| dd } n|dkrt| dkrxtdt | dd }|d	7 }| dd } nF|d
krt| dkrtdt | dd }|d7 }| dd } n|}|| fS )a   Read one to three bytes from 255UInt16-encoded input string, and return a
	tuple containing the decoded integer plus any leftover data.

	>>> unpack255UShort(bytechr(252))[0]
	252

	Note that some numbers (e.g. 506) can have multiple encodings:
	>>> unpack255UShort(struct.pack("BB", 254, 0))[0]
	506
	>>> unpack255UShort(struct.pack("BB", 255, 253))[0]
	506
	>>> unpack255UShort(struct.pack("BBB", 253, 1, 250))[0]
	506
	Nrm      r    z#not enough data to unpack 255UInt16z>H   r     r5  )r   r+   r   r   r,   )r?   rP  rO  rB   rB   rC   r    s*    r  c                 C   sr   | dk s| dkrt d| dk r,td| S | dk rFtdd| d S | d	k r`tdd
| d S tdd| S dS )z Encode unsigned integer in range 0 to 65535 (inclusive) to a bytestring
	using 255UInt16 variable-length encoding.

	>>> pack255UShort(252) == b'\xfc'
	True
	>>> pack255UShort(506) == b'\xfe\x00'
	True
	>>> pack255UShort(762) == b'\xfd\x02\xfa'
	True
	r   i  z/255UInt16 format requires 0 <= integer <= 65535rT  r:  rV  z>BBr5  i  rU  z>BHN)r   r   r   )r   rB   rB   rC   r2  0  s    r2  c                 C   sP   t d| |f  t| ddd}d|_|dur>t|j|d|_|j|dd dS )a   Compress OpenType font to WOFF2.

	Args:
		input_file: a file path, file or file-like object (open in binary mode)
			containing an OpenType font (either CFF- or TrueType-flavored).
		output_file: a file path, file or file-like object where to save the
			compressed WOFF2 font.
		transform_tables: Optional[Iterable[str]]: a set of table tags for which
			to enable preprocessing transformations. By default, only 'glyf'
			and 'loca' tables are transformed. An empty set means disable all
			transformations.
	Processing %s => %sFr!   r   N)r?   r[   ZreorderTables)r#   infor   rf   r:   r;   save)
input_fileoutput_filetransform_tablesfontrB   rB   rC   r}   H  s    r}   c                 C   s>   t d| |f  t| ddd}d|_d|_|j|dd dS )a  Decompress WOFF2 font to OpenType font.

	Args:
		input_file: a file path, file or file-like object (open in binary mode)
			containing a compressed WOFF2 font.
		output_file: a file path, file or file-like object where to save the
			decompressed OpenType font.
	rW  Fr!   NTrX  )r#   rY  r   rf   r;   rZ  )r[  r\  r^  rB   rB   rC   r7   b  s
    	r7   c              
      s  ddl  ddlm} ddlm} G  fddd j}G dd d j}G d	d
 d
 j} jdtj	dd}|j
dd|dd |jdd}|jddd}|jddd}	||	fD ]4}
|
jdd}|j
ddddd |j
dddd d q|j
d!d"d#d$ |	j
d!d"d%d$ |j
d&d'd(d)d$ |	j
d&d'd(d*d$ | }|j
d+d,d|d-d. |j
d/d,d|d0d. |jtd1d2hd3 |	jtd4 t|| }|d5d}|s|  dS |d6}|d7}||rd8n|rd9nd:d; |d< s|tu rd=}n|tu rft|d! d>$}|d? |d?}W d   n1 s20    Y  t|d?ksRJ d@|dAkr`dBndC}nt|||d! d|dD|d<< z|f i | W n0 ty } z|| W Y d}~n
d}~0 0 dS )Ez#Compress and decompress WOFF2 fontsr   N)configLogger)makeOutputFileNamec                       s   e Zd Zd fdd	ZdS )zmain.<locals>._HelpActionNc           	         sJ    fdd|j D }|D ]$}|j D ]\}}t|  q&q|  d S )Nc                    s   g | ]}t | jr|qS rB   )
isinstance_SubParsersAction)r   actionargparserB   rC   r   |  s   z6main.<locals>._HelpAction.__call__.<locals>.<listcomp>)_actionschoicesrz   printformat_helpexit)	r=   parser	namespacer   option_stringZsubparsers_actionsZsubparsers_actionchoice	subparserrd  rB   rC   __call__{  s    
z"main.<locals>._HelpAction.__call__)Nrc   rd   re   rp  rB   rd  rB   rC   _HelpActiony  s   rr  c                   @   s   e Zd ZdddZdS )z$main.<locals>._NoGlyfTransformActionNc                 S   s   |j ddh d S )NrJ   rL   )r]  difference_updater=   rk  rl  r   rm  rB   rB   rC   rp    s    z-main.<locals>._NoGlyfTransformAction.__call__)Nrq  rB   rB   rB   rC   _NoGlyfTransformAction  s   ru  c                   @   s   e Zd ZdddZdS )z"main.<locals>._HmtxTransformActionNc                 S   s   |j d d S r9  )r]  addrt  rB   rB   rC   rp    s    z+main.<locals>._HmtxTransformAction.__call__)Nrq  rB   rB   rB   rC   _HmtxTransformAction  s   rw  zfonttools ttLib.woff2F)progdescriptionadd_helpz-hz--helpzshow this help message and exit)rc  helpzsub-commands)titler}   z#Compress a TTF or OTF font to WOFF2)ry  r7   zDecompress a WOFF2 font to OTF)requiredz-vz	--verbose
store_truezprint more messages to consolez-qz--quietz do not print messages to consoler[  ZINPUTz&the input OpenType font (.ttf or .otf))metavarr{  zthe input WOFF2 fontz-oz--output-fileZOUTPUTzthe output WOFF2 fontzthe output OpenType fontz--no-glyf-transformr]  z'Do not transform glyf (and loca) tables)destnargsrc  r{  z--hmtx-transformz/Enable optional transformation for 'hmtx' tablerJ   rL   )
subcommandr]  )r  r  quietverboseERRORDEBUGINFO)levelr\  z.woff2rbr   znot enough datas   OTTOz.otfz.ttf)Z	outputDir	extension)re  Z	fontToolsr_  ZfontTools.ttxr`  rr  ActionArgumentParsermainr   add_argumentadd_subparsers
add_parseradd_mutually_exclusive_groupadd_argument_groupset_defaultsr}   r7   vars
parse_argspop
print_helpopenr(   r'   r+   AssertionErrorr   r$   )argsr_  r`  rr  ru  rw  rk  Zparser_groupZparser_compressZparser_decompressro  groupZtransform_groupoptionsr  r  r  r  fri   erB   rd  rC   r  s  s    






*

r  __main__)N)N)Lior   r   r   r   collectionsr   ZfontTools.miscr   ZfontTools.misc.arrayToolsr   ZfontTools.misc.textToolsr   r   r   r	   r
   ZfontTools.ttLibr   r   r   r   r   ZfontTools.ttLib.sfntr   r   r   r   r   r   r   r   r   ZfontTools.ttLib.tablesr   r   logging	getLoggerr#   r"   Z
brotlicffir6   r%   r   rg   r-   calcsizer*   r   r   Zwoff2FlagsSizer   r   r   rN  r   rL  r   r   r  rn   r)   rR   rS   r]   r:   r   rR  r   r  r2  r}   r7   r  rc   rj  rB   rB   rB   rC   <module>   sx   ,
   8
	


	L#  5 A-,

 	
