%PDF- %PDF-
| Direktori : /lib/python3/dist-packages/reportlab/platypus/ |
| Current File : //lib/python3/dist-packages/reportlab/platypus/tables.py |
#Copyright ReportLab Europe Ltd. 2000-2017
#see license.txt for license details
#history https://hg.reportlab.com/hg-public/reportlab/log/tip/src/reportlab/platypus/tables.py
__all__= (
'Table',
'TableStyle',
'CellStyle',
'LongTable',
)
__version__='3.5.21'
__doc__="""
Tables are created by passing the constructor a tuple of column widths, a tuple of row heights and the data in
row order. Drawing of the table can be controlled by using a TableStyle instance. This allows control of the
color and weight of the lines (if any), and the font, alignment and padding of the text.
None values in the sequence of row heights or column widths, mean that the corresponding rows
or columns should be automatically sized.
All the cell values should be convertible to strings; embedded newline '\\n' characters
cause the value to wrap (ie are like a traditional linefeed).
See the test output from running this module as a script for a discussion of the method for constructing
tables and table styles.
"""
from reportlab.platypus.flowables import Flowable, Preformatted
from reportlab import rl_config
from reportlab.lib.styles import PropertySet, ParagraphStyle, _baseFontName
from reportlab.lib import colors
from reportlab.lib.utils import annotateException, IdentStr, flatten, isStr, asNative, strTypes, __UNSET__
from reportlab.lib.validators import isListOfNumbersOrNone
from reportlab.lib.rl_accel import fp_str
from reportlab.lib.abag import ABag as CellFrame
from reportlab.pdfbase.pdfmetrics import stringWidth
from reportlab.platypus.doctemplate import Indenter, NullActionFlowable
from reportlab.platypus.flowables import LIIndenter
from collections import namedtuple
LINECAPS={None: None, 'butt':0,'round':1,'projecting':2,'squared':2}
LINEJOINS={None: None, 'miter':0, 'mitre':0, 'round':1,'bevel':2}
class CellStyle(PropertySet):
fontname = _baseFontName
fontsize = 10
leading = 12
leftPadding = 6
rightPadding = 6
topPadding = 3
bottomPadding = 3
firstLineIndent = 0
color = 'black'
alignment = 'LEFT'
background = 'white'
valign = "BOTTOM"
href = None
destination = None
def __init__(self, name, parent=None):
self.name = name
if parent is not None:
parent.copy(self)
def copy(self, result=None):
if result is None:
result = CellStyle(self.name)
for name in dir(self):
if name.startswith('_'): continue
setattr(result, name, getattr(self, name))
return result
class TableStyle:
def __init__(self, cmds=None, parent=None, **kw):
#handle inheritance from parent first.
if parent:
# copy the parents list at construction time
pcmds = parent.getCommands()[:]
self._opts = parent._opts
for a in ('spaceBefore','spaceAfter'):
if hasattr(parent,a):
setattr(self,a,getattr(parent,a))
else:
pcmds = []
self._cmds = pcmds + list(cmds or [])
self._opts={}
self._opts.update(kw)
def add(self, *cmd):
self._cmds.append(cmd)
def __repr__(self):
return "TableStyle(\n%s\n) # end TableStyle" % " \n".join(map(repr, self._cmds))
def getCommands(self):
return self._cmds
def _rowLen(x):
return not isinstance(x,(tuple,list)) and 1 or len(x)
def _calc_pc(V,avail):
'''check list V for percentage or * values
1) absolute values go through unchanged
2) percentages are used as weights for unconsumed space
3) if no None values were seen '*' weights are
set equally with unclaimed space
otherwise * weights are assigned as None'''
R = []
r = R.append
I = []
i = I.append
J = []
j = J.append
s = avail
w = n = 0.
for v in V:
if isinstance(v,strTypes):
v = str(v).strip()
if not v:
v = None
n += 1
elif v.endswith('%'):
v = float(v[:-1])
w += v
i(len(R))
elif v=='*':
j(len(R))
else:
v = float(v)
s -= v
elif v is None:
n += 1
else:
s -= v
r(v)
s = max(0.,s)
f = s/max(100.,w)
for i in I:
R[i] *= f
s -= R[i]
s = max(0.,s)
m = len(J)
if m:
v = n==0 and s/m or None
for j in J:
R[j] = v
return R
def _calcBezierPoints(P, kind):
'''calculate all or half of a bezier curve
kind==0 all, 1=first half else second half'''
if kind==0:
return P
else:
Q0 = (0.5*(P[0][0]+P[1][0]),0.5*(P[0][1]+P[1][1]))
Q1 = (0.5*(P[1][0]+P[2][0]),0.5*(P[1][1]+P[2][1]))
Q2 = (0.5*(P[2][0]+P[3][0]),0.5*(P[2][1]+P[3][1]))
R0 = (0.5*(Q0[0]+Q1[0]),0.5*(Q0[1]+Q1[1]))
R1 = (0.5*(Q1[0]+Q2[0]),0.5*(Q1[1]+Q2[1]))
S0 = (0.5*(R0[0]+R1[0]),0.5*(R0[1]+R1[1]))
return [P[0],Q0,R0,S0] if kind==1 else [S0,R1,Q2,P[3]]
def _quadrantDef(xpos, ypos, corner, r, kind=0, direction='left-right', m=0.4472):
t = m*r
if xpos=='right' and ypos=='bottom': #bottom right
xhi,ylo = corner
P = [(xhi - r, ylo),(xhi-t, ylo), (xhi, ylo + t), (xhi, ylo + r)]
elif xpos=='right' and ypos=='top': #top right
xhi,yhi = corner
P = [(xhi, yhi - r),(xhi, yhi - t), (xhi - t, yhi), (xhi - r, yhi)]
elif xpos=='left' and ypos=='top': #top left
xlo,yhi = corner
P = [(xlo + r, yhi),(xlo + t, yhi), (xlo, yhi - t), (xlo, yhi - r)]
elif xpos=='left' and ypos=='bottom': #bottom left
xlo,ylo = corner
P = [(xlo, ylo + r),(xlo, ylo + t), (xlo + t, ylo), (xlo + r, ylo)]
else:
raise ValueError(f'Unknown quadrant position (xpos,ypos)={(xpos,ypos)!r}')
if direction=='left-right' and P[0][0]>P[-1][0] or direction=='bottom-top' and P[0][1]>P[-1][1]:
P.reverse()
P = _calcBezierPoints(P, kind)
return P
def _hLine(canvLine, scp, ecp, y, hBlocks, FUZZ=rl_config._FUZZ):
'''
Draw horizontal lines; do not draw through regions specified in hBlocks
This also serves for vertical lines with a suitable canvLine
'''
if hBlocks: hBlocks = hBlocks.get(y,None)
if not hBlocks or scp>=hBlocks[-1][1]-FUZZ or ecp<=hBlocks[0][0]+FUZZ:
canvLine(scp,y,ecp,y)
else:
i = 0
n = len(hBlocks)
while scp<ecp-FUZZ and i<n:
x0, x1 = hBlocks[i]
if x1<=scp+FUZZ or x0>=ecp-FUZZ:
i += 1
continue
i0 = max(scp,x0)
i1 = min(ecp,x1)
if i0>scp: canvLine(scp,y,i0,y)
scp = i1
if scp<ecp-FUZZ: canvLine(scp,y,ecp,y)
def _multiLine(scp,ecp,y,canvLine,ws,count):
offset = 0.5*(count-1)*ws
y += offset
for idx in range(count):
canvLine(scp, y, ecp, y)
y -= ws
def _convert2int(value, map, low, high, name, cmd):
'''private converter tries map(value) low<=int(value)<=high or finally an error'''
try:
return map[value]
except KeyError:
try:
ivalue = int(value)
if low<=ivalue<=high: return ivalue
except:
pass
raise ValueError(f'Bad {name} value {value} in {cmd!a}')
def _endswith(obj,s):
try:
return obj.endswith(s)
except:
return 0
def spanFixDim(V0,V,spanCons,lim=None,FUZZ=rl_config._FUZZ):
#assign required space to variable rows equally to existing calculated values
M = {}
if not lim: lim = len(V0) #in longtables the row calcs may be truncated
#we assign the largest spaces first hoping to get a smaller result
for v,(x0,x1) in reversed(sorted(((iv,ik) for ik,iv in spanCons.items()))):
if x0>=lim: continue
x1 += 1
t = sum([V[x]+M.get(x,0) for x in range(x0,x1)])
if t>=v-FUZZ: continue #already good enough
X = [x for x in range(x0,x1) if V0[x] is None] #variable candidates
if not X: continue #something wrong here mate
v -= t
v /= float(len(X))
for x in X:
M[x] = M.get(x,0)+v
for x,v in M.items():
V[x] += v
class _ExpandedCellTuple(tuple):
pass
class _ExpandedCellTupleEx(tuple):
def __new__(cls,seq,tagType,altText,extras):
self = tuple.__new__(cls,seq)
self.tagType = tagType
self.altText = altText
self.extras = extras
return self
RoundingRectDef = namedtuple('RoundingRectDefs','x0 y0 w h x1 y1 ar SL')
RoundingRectLine = namedtuple('RoundingRectLine','xs ys xe ye weight color cap dash join')
_SPECIALROWS=("splitfirst", "splitlast", "inrowsplitstart","inrowsplitend")
class Table(Flowable):
def __init__(self, data, colWidths=None, rowHeights=None, style=None,
repeatRows=0, repeatCols=0, splitByRow=1, splitInRow=0, emptyTableAction=None, ident=None,
hAlign=None,vAlign=None, normalizedData=0, cellStyles=None, rowSplitRange=None,
spaceBefore=None,spaceAfter=None, longTableOptimize=None, minRowHeights=None,
cornerRadii=__UNSET__, #or [topLeft, topRight, bottomLeft bottomRight]
renderCB=None,
):
self.ident = ident
self.hAlign = hAlign or 'CENTER'
self.vAlign = vAlign or 'MIDDLE'
if not isinstance(data,(tuple,list)):
raise ValueError("%s invalid data type" % self.identity())
self._renderCB = renderCB
self._nrows = nrows = len(data)
self._cellvalues = []
_seqCW = isinstance(colWidths,(tuple,list))
_seqRH = isinstance(rowHeights,(tuple,list))
if nrows: self._ncols = ncols = max(list(map(_rowLen,data)))
elif colWidths and _seqCW: ncols = len(colWidths)
else: ncols = 0
if not emptyTableAction: emptyTableAction = rl_config.emptyTableAction
self._longTableOptimize = (getattr(self,'_longTableOptimize',rl_config.longTableOptimize)
if longTableOptimize is None else longTableOptimize)
if not (nrows and ncols):
if emptyTableAction=='error':
raise ValueError(f'{self.identity()} must have at least a row and column')
elif emptyTableAction=='indicate':
self.__class__ = Preformatted
global _emptyTableStyle
if '_emptyTableStyle' not in list(globals().keys()):
_emptyTableStyle = ParagraphStyle('_emptyTableStyle')
_emptyTableStyle.textColor = colors.red
_emptyTableStyle.backColor = colors.yellow
Preformatted.__init__(self,'%s(%d,%d)' % (self.__class__.__name__,nrows,ncols), _emptyTableStyle)
elif emptyTableAction=='ignore':
self.__class__ = NullActionFlowable
else:
raise ValueError(f'{self.identitiy()} bad emptyTableAction: {emptyTableAction!a}')
return
# we need a cleanup pass to ensure data is strings - non-unicode and non-null
if normalizedData:
self._cellvalues = data
else:
self._cellvalues = data = self.normalizeData(data)
if not _seqCW: colWidths = ncols*[colWidths]
elif len(colWidths)!=ncols:
if rl_config.allowShortTableRows and isinstance(colWidths,list):
n = len(colWidths)
if n<ncols:
colWidths[n:] = (ncols-n)*[colWidths[-1]]
else:
colWidths = colWidths[:ncols]
else:
raise ValueError(f'{self.identity()} data error - {ncols} columns in data but {len(colWidths)} column widths')
if not _seqRH: rowHeights = nrows*[rowHeights]
elif len(rowHeights) != nrows:
raise ValueError(f'{self.identity()} data error - {nrows} rows in data but {len(rowHeights)} row heights')
for i,d in enumerate(data):
n = len(d)
if n!=ncols:
if rl_config.allowShortTableRows and isinstance(d,list):
d[n:] = (ncols-n)*['']
else:
raise ValueError(f'{self.identity()} expected {ncols} not {n} columns in row {i}!')
self._rowHeights = self._argH = rowHeights
self._colWidths = self._argW = colWidths
if cellStyles is None:
cellrows = []
for i in range(nrows):
cellcols = []
for j in range(ncols):
cellcols.append(CellStyle(repr((i,j))))
cellrows.append(cellcols)
self._cellStyles = cellrows
else:
self._cellStyles = cellStyles
self._bkgrndcmds = []
self._linecmds = []
self._spanCmds = []
self._nosplitCmds = []
self._srflcmds = [] # split first last
self._sircmds = [] # split in row special commands
# NB repeatRows can be a list or tuple eg (1,) repeats only the second row of a table
# or an integer eg 2 to repeat both rows 0 & 1
self.repeatRows = repeatRows
self.repeatCols = repeatCols
self.splitByRow = splitByRow
self.splitInRow = splitInRow
if style:
self.setStyle(style)
if cornerRadii is not __UNSET__: #instance argument overrides
self._setCornerRadii(cornerRadii)
self._rowSplitRange = rowSplitRange
if spaceBefore is not None:
self.spaceBefore = spaceBefore
if spaceAfter is not None:
self.spaceAfter = spaceAfter
if minRowHeights != None:
lmrh = len(minRowHeights)
if not lmrh:
raise ValueError('{self.idenity()} Supplied mismatching minimum row heights of length {lmrh}')
elif lmrh<nrows:
minRowHeights = minRowHeights+(nrows-lmrh)*minRowHeights.__class__((0,))
self._minRowHeights = minRowHeights
def __repr__(self):
"incomplete, but better than nothing"
r = getattr(self,'_rowHeights','[unknown]')
c = getattr(self,'_colWidths','[unknown]')
cv = getattr(self,'_cellvalues','[unknown]')
import pprint
cv = pprint.pformat(cv)
cv = cv.replace("\n", "\n ")
return "%s(\n rowHeights=%s,\n colWidths=%s,\n%s\n) # end table" % (self.__class__.__name__,r,c,cv)
def normalizeData(self, data):
"""Takes a block of input data (list of lists etc.) and
- coerces unicode strings to non-unicode UTF8
- coerces nulls to ''
-
"""
def normCell(stuff):
if stuff is None:
return ''
elif isStr(stuff):
return asNative(stuff)
else:
return stuff
outData = []
for row in data:
outRow = [normCell(cell) for cell in row]
outData.append(outRow)
return outData
def identity(self, maxLen=30):
'''Identify our selves as well as possible'''
if self.ident: return self.ident
vx = None
nr = getattr(self,'_nrows','unknown')
nc = getattr(self,'_ncols','unknown')
cv = getattr(self,'_cellvalues',None)
rh = getattr(self, '_rowHeights', None)
if cv and 'unknown' not in (nr,nc):
b = 0
for i in range(nr):
for j in range(nc):
v = cv[i][j]
if isinstance(v,(list,tuple,Flowable)):
if not isinstance(v,(tuple,list)): v = (v,)
r = ''
for vij in v:
r = vij.identity(maxLen)
if r and r[-4:]!='>...':
break
if r and r[-4:]!='>...':
ix, jx, vx, b = i, j, r, 1
else:
v = v is None and '' or str(v)
ix, jx, vx = i, j, v
b = (vx and isinstance(v,strTypes)) and 1 or 0
if maxLen: vx = vx[:maxLen]
if b: break
if b: break
if rh: #find tallest row, it's of great interest'
tallest = '(tallest row %d)' % int(max(rh))
else:
tallest = ''
if vx:
vx = ' with cell(%d,%d) containing\n%s' % (ix,jx,repr(vx))
else:
vx = '...'
return "<%s@0x%8.8X %s rows x %s cols%s>%s" % (self.__class__.__name__, id(self), nr, nc, tallest, vx)
def _cellListIter(self,C,aW,aH):
canv = getattr(self,'canv',None)
for c in C:
if getattr(c,'__split_only__',None):
for d in c.splitOn(canv,aW,aH):
yield d
else:
yield c
def _cellListProcess(self,v,aW,aH):
if isinstance(v,_ExpandedCellTuple):
C = v
else:
C = (v,) if isinstance(v,Flowable) else flatten(v)
frame = None
R = [].append
for c in self._cellListIter(C,aW,aH):
if isinstance(c,Indenter):
if not frame:
frame = CellFrame(_leftExtraIndent=0,_rightExtraIndent=0)
c.frameAction(frame)
if frame._leftExtraIndent<1e-8 and frame._rightExtraIndent<1e-8:
frame = None
continue
if frame:
R(LIIndenter(c,leftIndent=frame._leftExtraIndent,rightIndent=frame._rightExtraIndent))
else:
R(c)
if hasattr(v,'tagType'):
C = _ExpandedCellTupleEx(R.__self__,v.tagType,v.altText,v.extras)
else:
C = _ExpandedCellTuple(R.__self__)
return C
def _listCellGeom(self, V,w,s,W=None,H=None,aH=72000):
if not V: return 0,0
aW = w - s.leftPadding - s.rightPadding
aH = aH - s.topPadding - s.bottomPadding
t = 0
w = 0
canv = getattr(self,'canv',None)
sb0 = None
if isinstance(V, str):
vw = self._elementWidth(V, s)
vh = len(V.split('\n'))*s.fontsize*1.2
return max(w, vw), vh
for v in V:
vw, vh = v.wrapOn(canv, aW, aH)
sb = v.getSpaceBefore()
sa = v.getSpaceAfter()
if W is not None: W.append(vw)
if H is not None: H.append(vh)
w = max(w,vw)
t += vh + sa + sb
if sb0 is None:
sb0 = sb
return w, t - sb0 - sa
def _listValueWidth(self,V,aH=72000,aW=72000):
if not V: return 0,0
t = 0
w = 0
canv = getattr(self,'canv',None)
return max([v.wrapOn(canv,aW,aH)[0] for v in V])
def _calc_width(self,availWidth,W=None):
if getattr(self,'_width_calculated_once',None): return
#comments added by Andy to Robin's slightly terse variable names
if not W: W = _calc_pc(self._argW,availWidth) #widths array
if None in W: #some column widths are not given
canv = getattr(self,'canv',None)
saved = None
if self._spanCmds:
colSpanCells = self._colSpanCells
spanRanges = self._spanRanges
else:
colSpanCells = ()
spanRanges = {}
spanCons = {}
if W is self._argW:
W0 = W
W = W[:]
else:
W0 = W[:]
V = self._cellvalues
S = self._cellStyles
while None in W:
j = W.index(None) #find first unspecified column
w = 0
for i,Vi in enumerate(V):
v = Vi[j]
s = S[i][j]
ji = j,i
span = spanRanges.get(ji,None)
if ji in colSpanCells and not span: #if the current cell is part of a spanned region,
t = 0.0 #assume a zero size.
else:#work out size
t = self._elementWidth(v,s)
if t is None:
raise ValueError(f'Flowable {v.identity()} in cell({i},{j}) can\'t have auto width\n{self.identity(30)}')
t += s.leftPadding+s.rightPadding
if span:
c0 = span[0]
c1 = span[2]
if c0!=c1:
x = c0,c1
spanCons[x] = max(spanCons.get(x,t),t)
t = 0
if t>w: w = t #record a new maximum
W[j] = w
if spanCons:
try:
spanFixDim(W0,W,spanCons)
except:
annotateException('\nspanning problem in %s\nW0=%r W=%r\nspanCons=%r' % (self.identity(),W0,W,spanCons))
self._colWidths = W
width = 0
self._colpositions = [0] #index -1 is right side boundary; we skip when processing cells
for w in W:
width = width + w
self._colpositions.append(width)
self._width = width
self._width_calculated_once = 1
def _elementWidth(self,v,s):
if isinstance(v,(list,tuple)):
w = 0
for e in v:
ew = self._elementWidth(e,s)
if ew is None: return None
w = max(w,ew)
return w
elif isinstance(v,Flowable):
if v._fixedWidth:
if hasattr(v, 'width') and isinstance(v.width,(int,float)): return v.width
if hasattr(v, 'drawWidth') and isinstance(v.drawWidth,(int,float)): return v.drawWidth
if hasattr(v,'__styledWrap__'): #very experimental
try:
return getattr(v,'__styledWrap__')(s)[0]
except:
pass
# Even if something is fixedWidth, the attribute to check is not
# necessarily consistent (cf. Image.drawWidth). Therefore, we'll
# be extra-careful and fall through to this code if necessary.
if hasattr(v, 'minWidth'):
try:
w = v.minWidth() # should be all flowables
if isinstance(w,(float,int)): return w
except AttributeError:
pass
if v is None:
return 0
else:
try:
v = str(v).split("\n")
except:
return 0
fontName = s.fontname
fontSize = s.fontsize
return max([stringWidth(x,fontName,fontSize) for x in v])
def _calc_height(self, availHeight, availWidth, H=None, W=None):
H = self._argH
if not W: W = _calc_pc(self._argW,availWidth) #widths array
hmax = lim = len(H)
longTable = self._longTableOptimize
if None in H:
minRowHeights = self._minRowHeights
canv = getattr(self,'canv',None)
saved = None
#get a handy list of any cells which span rows. should be ignored for sizing
if self._spanCmds:
rowSpanCells = self._rowSpanCells
colSpanCells = self._colSpanCells
spanRanges = self._spanRanges
colpositions = self._colpositions
else:
rowSpanCells = colSpanCells = ()
spanRanges = {}
if canv: saved = canv._fontname, canv._fontsize, canv._leading
H0 = H
H = H[:] #make a copy as we'll change it
self._rowHeights = H
spanCons = {}
FUZZ = rl_config._FUZZ
while None in H:
i = H.index(None)
V = self._cellvalues[i] # values for row i
S = self._cellStyles[i] # styles for row i
h = 0
j = 0
for j,(v, s, w) in enumerate(list(zip(V, S, W))): # value, style, width (lengths must match)
ji = j,i
span = spanRanges.get(ji,None)
if ji in rowSpanCells and not span:
continue # don't count it, it's either occluded or unreliable
else:
if isinstance(v,(tuple,list,Flowable)):
v = V[j] = self._cellListProcess(v,w,None)
if w is None and not self._canGetWidth(v):
raise ValueError(f'Flowable {v[0].identity()} in cell({i},{j}) can\'t have auto width\n{self.identity(30)}')
if canv: canv._fontname, canv._fontsize, canv._leading = s.fontname, s.fontsize, s.leading or 1.2*s.fontsize
if ji in colSpanCells:
if not span: continue
w = max(colpositions[span[2]+1]-colpositions[span[0]],w or 0)
dW,t = self._listCellGeom(v,w or self._listValueWidth(v),s)
if canv: canv._fontname, canv._fontsize, canv._leading = saved
dW = dW + s.leftPadding + s.rightPadding
if not rl_config.allowTableBoundsErrors and dW>w:
from reportlab.platypus.doctemplate import LayoutError
raise LayoutError("Flowable %s (%sx%s points) too wide for cell(%d,%d) (%sx* points) in\n%s" % (v[0].identity(30),fp_str(dW),fp_str(t),i,j, fp_str(w), self.identity(30)))
else:
v = (v is not None and str(v) or '').split("\n")
t = (s.leading or 1.2*s.fontsize)*len(v)
t += s.bottomPadding+s.topPadding
if span:
r0 = span[1]
r1 = span[3]
if r0!=r1:
x = r0,r1
spanCons[x] = max(spanCons.get(x,t),t)
t = 0
if t>h: h = t #record a new maximum
# If a minimum height has been specified use that, otherwise allow the cell to grow
H[i] = max(minRowHeights[i],h) if minRowHeights else h
# we can stop if we have filled up all available room
if longTable:
hmax = i+1 #we computed H[i] so known len == i+1
height = sum(H[:hmax])
if height > availHeight:
#we can terminate if all spans are complete in H[:hmax]
if spanCons:
msr = max(x[1] for x in spanCons.keys()) #RS=[endrowspan,.....]
if hmax>msr:
break
if None not in H: hmax = lim
if spanCons:
try:
spanFixDim(H0,H,spanCons)
except:
annotateException('\nspanning problem in %s hmax=%s lim=%s avail=%s x %s\nH0=%r H=%r\nspanCons=%r' % (self.identity(),hmax,lim,availWidth,availHeight,H0,H,spanCons))
#iterate backwards through the heights to get rowpositions in reversed order
self._rowpositions = j = []
height = c = 0
for i in range(hmax-1,-1,-1):
j.append(height)
y = H[i] - c
t = height + y
c = (t - height) - y
height = t
j.append(height)
self._height = height
j.reverse() #reverse the reversed list of row positions
self._hmax = hmax
def _calc(self, availWidth, availHeight):
#if hasattr(self,'_width'): return
#in some cases there are unsizable things in
#cells. If so, apply a different algorithm
#and assign some withs in a less (thanks to Gary Poster) dumb way.
#this CHANGES the widths array.
if (None in self._colWidths or '*' in self._colWidths) and self._hasVariWidthElements():
W = self._calcPreliminaryWidths(availWidth) #widths
else:
W = None
# need to know which cells are part of spanned
# ranges, so _calc_height and _calc_width can ignore them
# in sizing
if self._spanCmds:
self._calcSpanRanges()
if None in self._argH:
self._calc_width(availWidth,W=W)
if self._nosplitCmds:
self._calcNoSplitRanges()
# calculate the full table height
self._calc_height(availHeight,availWidth,W=W)
# calculate the full table width
self._calc_width(availWidth,W=W)
if self._spanCmds:
#now work out the actual rect for each spanned cell from the underlying grid
self._calcSpanRects()
def _culprit(self):
"""Return a string describing the tallest element.
Usually this is what causes tables to fail to split. Currently
tables are the only items to have a '_culprit' method. Doctemplate
checks for it.
"""
rh = self._rowHeights
tallest = max(rh)
rowNum = rh.index(tallest)
#rowNum of limited interest as usually it's a split one
#and we see row #1. Text might be a nice addition.
return 'tallest cell %0.1f points' % tallest
def _hasVariWidthElements(self, upToRow=None):
"""Check for flowables in table cells and warn up front.
Allow a couple which we know are fixed size such as
images and graphics."""
if upToRow is None: upToRow = self._nrows
for row in range(min(self._nrows, upToRow)):
for col in range(self._ncols):
value = self._cellvalues[row][col]
if not self._canGetWidth(value):
return 1
return 0
def _canGetWidth(self, thing):
"Can we work out the width quickly?"
if isinstance(thing,(list, tuple)):
for elem in thing:
if not self._canGetWidth(elem):
return 0
return 1
elif isinstance(thing, Flowable):
return thing._fixedWidth # must loosen this up
else: #str, number, None etc.
#anything else gets passed to str(...)
# so should be sizable
return 1
def _calcPreliminaryWidths(self, availWidth):
"""Fallback algorithm for when main one fails.
Where exact width info not given but things like
paragraphs might be present, do a preliminary scan
and assign some best-guess values."""
W = list(self._argW) # _calc_pc(self._argW,availWidth)
#_debug = getattr(self,'_debug',0)
totalDefined = 0.0
percentDefined = 0
percentTotal = 0
numberUndefined = 0
numberGreedyUndefined = 0
for w in W:
if w is None:
numberUndefined += 1
elif w == '*':
numberUndefined += 1
numberGreedyUndefined += 1
elif _endswith(w,'%'):
percentDefined += 1
percentTotal += float(w[:-1])
else:
assert isinstance(w,(int,float))
totalDefined = totalDefined + w
#if _debug: print('prelim width calculation. %d columns, %d undefined width, %0.2f units remain' % (self._ncols, numberUndefined, availWidth - totalDefined))
#check columnwise in each None column to see if they are sizable.
given = []
sizeable = []
unsizeable = []
minimums = {}
totalMinimum = 0
elementWidth = self._elementWidth
for colNo in range(self._ncols):
w = W[colNo]
if w is None or w=='*' or _endswith(w,'%'):
siz = 1
final = 0
for rowNo in range(self._nrows):
value = self._cellvalues[rowNo][colNo]
style = self._cellStyles[rowNo][colNo]
new = elementWidth(value,style) or 0
new += style.leftPadding+style.rightPadding
#if _debug: print('[%d,%d] new=%r-->%r' % (rowNo,colNo,new - style.leftPadding+style.rightPadding, new))
final = max(final, new)
siz = siz and self._canGetWidth(value) # irrelevant now?
if siz:
sizeable.append(colNo)
else:
unsizeable.append(colNo)
minimums[colNo] = final
totalMinimum += final
else:
given.append(colNo)
if len(given) == self._ncols:
return
#if _debug: print('predefined width: ',given)
#if _debug: print('uncomputable width: ',unsizeable)
#if _debug: print('computable width: ',sizeable)
#if _debug: print('minimums=%r' % (list(sorted(list(minimums.items()))),))
# how much width is left:
remaining = availWidth - (totalMinimum + totalDefined)
if remaining > 0:
# we have some room left; fill it.
definedPercentage = (totalDefined/float(availWidth))*100
percentTotal += definedPercentage
if numberUndefined and percentTotal < 100:
undefined = numberGreedyUndefined or numberUndefined
defaultWeight = (100-percentTotal)/float(undefined)
percentTotal = 100
defaultDesired = (defaultWeight/float(percentTotal))*availWidth
else:
defaultWeight = defaultDesired = 1
# we now calculate how wide each column wanted to be, and then
# proportionately shrink that down to fit the remaining available
# space. A column may not shrink less than its minimum width,
# however, which makes this a bit more complicated.
desiredWidths = []
totalDesired = 0
effectiveRemaining = remaining
for colNo, minimum in minimums.items():
w = W[colNo]
if _endswith(w,'%'):
desired = (float(w[:-1])/percentTotal)*availWidth
elif w == '*':
desired = defaultDesired
else:
desired = not numberGreedyUndefined and defaultDesired or 1
if desired <= minimum:
W[colNo] = minimum
else:
desiredWidths.append(
(desired-minimum, minimum, desired, colNo))
totalDesired += desired
effectiveRemaining += minimum
if desiredWidths: # else we're done
# let's say we have two variable columns. One wanted
# 88 points, and one wanted 264 points. The first has a
# minWidth of 66, and the second of 55. We have 71 points
# to divide up in addition to the totalMinimum (i.e.,
# remaining==71). Our algorithm tries to keep the proportion
# of these variable columns.
#
# To do this, we add up the minimum widths of the variable
# columns and the remaining width. That's 192. We add up the
# totalDesired width. That's 352. That means we'll try to
# shrink the widths by a proportion of 192/352--.545454.
# That would make the first column 48 points, and the second
# 144 points--adding up to the desired 192.
#
# Unfortunately, that's too small for the first column. It
# must be 66 points. Therefore, we go ahead and save that
# column width as 88 points. That leaves (192-88==) 104
# points remaining. The proportion to shrink the remaining
# column is (104/264), which, multiplied by the desired
# width of 264, is 104: the amount assigned to the remaining
# column.
proportion = effectiveRemaining/float(totalDesired)
# we sort the desired widths by difference between desired and
# and minimum values, a value called "disappointment" in the
# code. This means that the columns with a bigger
# disappointment will have a better chance of getting more of
# the available space.
desiredWidths.sort()
finalSet = []
for disappointment, minimum, desired, colNo in desiredWidths:
adjusted = proportion * desired
if adjusted < minimum:
W[colNo] = minimum
totalDesired -= desired
effectiveRemaining -= minimum
if totalDesired:
proportion = effectiveRemaining/float(totalDesired)
else:
finalSet.append((minimum, desired, colNo))
for minimum, desired, colNo in finalSet:
adjusted = proportion * desired
assert adjusted >= minimum
W[colNo] = adjusted
else:
if percentTotal>0:
#transfer percentages to the defined cols
d = []
for colNo, w in minimums.items():
if w.endswith('%'):
W[colNo] = w = availWidth*float(w[:-1])/percentTotal
totalDefined += w
d.append(colNo)
for colNo in d:
del minimums[colNo]
del d
totalMinimum = sum(minimums.values())
remaining = availWidth - (totalDefined + totalMinimum)
#if _debug: print(f'0:remaining={remaining} totalDefined={totalDefined}')
#if _debug: print(f'0:minimums={minimums}')
if remaining<0 and totalDefined*rl_config.defCWRF+remaining>=0:
adj = -remaining/totalDefined
for colNo, w in enumerate(W):
if colNo not in minimums:
dw = adj*w
W[colNo] -= dw
totalDefined -= dw
remaining = availWidth - (totalDefined + totalMinimum)
#if _debug: print(f'1:remaining={remaining} totalDefined={totalDefined}')
#if _debug: print(f'1:minimums={minimums}')
adj = 1
else:
remaining = availWidth - totalDefined
adj = 1 if remaining<=0 else remaining/totalMinimum
for colNo, minimum in minimums.items():
W[colNo] = minimum * adj
#if _debug: print(f'new widths are: {W} sum={sum(W)} availWidth={availWidth}')
self._argW = self._colWidths = W
return W
def minWidth(self):
W = list(self._argW)
width = 0
elementWidth = self._elementWidth
rowNos = range(self._nrows)
values = self._cellvalues
styles = self._cellStyles
for colNo in range(len(W)):
w = W[colNo]
if w is None or w=='*' or _endswith(w,'%'):
final = 0
for rowNo in rowNos:
value = values[rowNo][colNo]
style = styles[rowNo][colNo]
new = (elementWidth(value,style)+
style.leftPadding+style.rightPadding)
final = max(final, new)
width += final
else:
width += float(w)
return width # XXX + 1/2*(left and right border widths)
def _calcSpanRanges(self):
"""Work out rects for tables which do row and column spanning.
This creates some mappings to let the later code determine
if a cell is part of a "spanned" range.
self._spanRanges shows the 'coords' in integers of each
'cell range', or None if it was clobbered:
(col, row) -> (col0, row0, col1, row1)
Any cell not in the key is not part of a spanned region
"""
self._spanRanges = spanRanges = {}
for x in range(self._ncols):
for y in range(self._nrows):
spanRanges[x,y] = (x, y, x, y)
self._colSpanCells = []
self._rowSpanCells = []
csa = self._colSpanCells.append
rsa = self._rowSpanCells.append
for (cmd, start, stop) in self._spanCmds:
x0, y0 = start
x1, y1 = stop
#normalize
if x0 < 0: x0 = x0 + self._ncols
if x1 < 0: x1 = x1 + self._ncols
if y0 < 0: y0 = y0 + self._nrows
if y1 < 0: y1 = y1 + self._nrows
if x0 > x1: x0, x1 = x1, x0
if y0 > y1: y0, y1 = y1, y0
if x0!=x1 or y0!=y1:
if x0!=x1: #column span
for y in range(y0, y1+1):
for x in range(x0,x1+1):
csa((x,y))
if y0!=y1: #row span
for y in range(y0, y1+1):
for x in range(x0,x1+1):
rsa((x,y))
for y in range(y0, y1+1):
for x in range(x0,x1+1):
spanRanges[x,y] = None
# set the main entry
spanRanges[x0,y0] = (x0, y0, x1, y1)
def _calcNoSplitRanges(self):
"""
This creates some mappings to let the later code determine
if a cell is part of a "nosplit" range.
self._nosplitRanges shows the 'coords' in integers of each
'cell range', or None if it was clobbered:
(col, row) -> (col0, row0, col1, row1)
Any cell not in the key is not part of a spanned region
"""
self._nosplitRanges = nosplitRanges = {}
for x in range(self._ncols):
for y in range(self._nrows):
nosplitRanges[x,y] = (x, y, x, y)
self._colNoSplitCells = []
self._rowNoSplitCells = []
csa = self._colNoSplitCells.append
rsa = self._rowNoSplitCells.append
for (cmd, start, stop) in self._nosplitCmds:
x0, y0 = start
x1, y1 = stop
#normalize
if x0 < 0: x0 = x0 + self._ncols
if x1 < 0: x1 = x1 + self._ncols
if y0 < 0: y0 = y0 + self._nrows
if y1 < 0: y1 = y1 + self._nrows
if x0 > x1: x0, x1 = x1, x0
if y0 > y1: y0, y1 = y1, y0
if x0!=x1 or y0!=y1:
#column span
if x0!=x1:
for y in range(y0, y1+1):
for x in range(x0,x1+1):
csa((x,y))
#row span
if y0!=y1:
for y in range(y0, y1+1):
for x in range(x0,x1+1):
rsa((x,y))
for y in range(y0, y1+1):
for x in range(x0,x1+1):
nosplitRanges[x,y] = None
# set the main entry
nosplitRanges[x0,y0] = (x0, y0, x1, y1)
def _calcSpanRects(self):
"""Work out rects for tables which do row and column spanning.
Based on self._spanRanges, which is already known,
and the widths which were given or previously calculated,
self._spanRects shows the real coords for drawing:
(col, row) -> (x, y, width, height)
for each cell. Any cell which 'does not exist' as another
has spanned over it will get a None entry on the right
"""
spanRects = getattr(self,'_spanRects',{})
hmax = getattr(self,'_hmax',None)
longTable = self._longTableOptimize
if spanRects and (longTable and hmax==self._hmax_spanRects or not longTable):
return
colpositions = self._colpositions
rowpositions = self._rowpositions
vBlocks = {}
hBlocks = {}
rlim = len(rowpositions)-1
for (coord, value) in self._spanRanges.items():
if value is None:
spanRects[coord] = None
else:
try:
col0, row0, col1, row1 = value
if row1>=rlim: continue
col,row = coord
if col1-col0>0:
for _ in range(col0+1,col1+1):
vBlocks.setdefault(colpositions[_],[]).append((rowpositions[row1+1],rowpositions[row0]))
if row1-row0>0:
for _ in range(row0+1,row1+1):
hBlocks.setdefault(rowpositions[_],[]).append((colpositions[col0],colpositions[col1+1]))
x = colpositions[col0]
y = rowpositions[row1+1]
width = colpositions[col1+1] - x
height = rowpositions[row0] - y
spanRects[coord] = (x, y, width, height)
except:
annotateException('\nspanning problem in %s' % (self.identity(),))
for _ in hBlocks, vBlocks:
for value in _.values():
value.sort()
self._spanRects = spanRects
self._vBlocks = vBlocks
self._hBlocks = hBlocks
self._hmax_spanRects = hmax
def setStyle(self, tblstyle):
if not isinstance(tblstyle,TableStyle):
tblstyle = TableStyle(tblstyle)
for cmd in tblstyle.getCommands():
if len(cmd)>=3:
c, (sc,sr), (ec,er) = cmd[0:3]
if (isinstance(sc,str) or isinstance(ec,str)
or (isinstance(sr,str) and sr not in _SPECIALROWS)
or (isinstance(er,str) and er not in _SPECIALROWS)):
raise ValueError(f'''bad style command {cmd!r} illegal of invalid string coordinate
only rows may be strings with values in {_SPECIALROWS!r}''')
self._addCommand(cmd)
for k,v in tblstyle._opts.items():
setattr(self,k,v)
for a in ('spaceBefore','spaceAfter'):
if not hasattr(self,a) and hasattr(tblstyle,a):
setattr(self,a,getattr(tblstyle,a))
def normCellRange(self, sc, ec, sr, er):
'''ensure cell range ends are with the table bounds'''
if sc < 0: sc = sc + self._ncols
if ec < 0: ec = ec + self._ncols
if sr < 0: sr = sr + self._nrows
if er < 0: er = er + self._nrows
return max(0,sc), min(self._ncols-1,ec), max(0,sr), min(self._nrows-1,er)
def _addCommand(self,cmd):
if cmd[0] in ('BACKGROUND','ROWBACKGROUNDS','COLBACKGROUNDS'):
self._bkgrndcmds.append(cmd)
elif cmd[0] == 'SPAN':
self._spanCmds.append(cmd)
elif cmd[0] == 'NOSPLIT':
# we expect op, start, stop
self._nosplitCmds.append(cmd)
elif _isLineCommand(cmd):
# we expect op, start, stop, weight, colour, cap, dashes, join
cmd = list(cmd)
if len(cmd)<5: raise ValueError(f'bad line command {cmd!a}')
#determine line cap value at position 5. This can be str or numeric.
if len(cmd)<6:
cmd.append(1)
else:
cap = _convert2int(cmd[5], LINECAPS, 0, 2, 'cap', cmd)
cmd[5] = cap
#dashes at index 6 - this is a dash array:
if len(cmd)<7: cmd.append(None)
#join mode at index 7 - can be str or numeric, look up as for caps
if len(cmd)<8: cmd.append(1)
else:
join = _convert2int(cmd[7], LINEJOINS, 0, 2, 'join', cmd)
cmd[7] = join
#linecount at index 8. Default is 1, set to 2 for double line.
if len(cmd)<9: cmd.append(1)
else:
lineCount = cmd[8]
if lineCount is None:
lineCount = 1
cmd[8] = lineCount
assert lineCount >= 1
#linespacing at index 9. Not applicable unless 2+ lines, defaults to line
#width so you get a visible gap between centres
if len(cmd)<10: cmd.append(cmd[3])
else:
space = cmd[9]
if space is None:
space = cmd[3]
cmd[9] = space
assert len(cmd) == 10
self._linecmds.append(tuple(cmd))
elif cmd[0]=="ROUNDEDCORNERS":
self._setCornerRadii(cmd[1])
else:
(op, (sc, sr), (ec, er)), values = cmd[:3] , cmd[3:]
if sr in _SPECIALROWS:
(self._srflcmds if sr[0]=='s' else self._sircmds).append(cmd)
else:
sc, ec, sr, er = self.normCellRange(sc,ec,sr,er)
ec += 1
for i in range(sr, er+1):
for j in range(sc, ec):
_setCellStyle(self._cellStyles, i, j, op, values)
def _drawLines(self):
ccap, cdash, cjoin = None, None, None
canv = self.canv
canv.saveState()
rrd = self._roundingRectDef
if rrd: #we are collection some lines
SL = rrd.SL
SL[:] = [] #empty saved lines list
ocanvline = canv.line
aSL = SL.append
def rcCanvLine(xs, ys, xe, ye):
if (
(xs==xe and (xs>=rrd.x1 or xs<=rrd.x0)) #vertical line that needs to be saved
or
(ys==ye and (ys>=rrd.y1 or ys<=rrd.y0)) #horizontal line that needs to be saved
):
aSL(RoundingRectLine(xs,ys,xe,ye,weight,color,cap,dash,join))
else:
ocanvline(xs,ys,xe,ye)
canv.line = rcCanvLine
try:
for op, (sc,sr), (ec,er), weight, color, cap, dash, join, count, space in self._linecmds:
if isinstance(sr,strTypes) and sr in _SPECIALROWS: continue
if cap!=None and ccap!=cap:
canv.setLineCap(cap)
ccap = cap
if dash is None or dash == []:
if cdash is not None:
canv.setDash()
cdash = None
elif dash != cdash:
canv.setDash(dash)
cdash = dash
if join is not None and cjoin!=join:
canv.setLineJoin(join)
cjoin = join
sc, ec, sr, er = self.normCellRange(sc,ec,sr,er)
getattr(self,_LineOpMap.get(op, '_drawUnknown' ))( (sc, sr), (ec, er), weight, color, count, space)
finally:
if rrd:
canv.line = ocanvline
canv.restoreState()
self._curcolor = None
def _drawUnknown(self, start, end, weight, color, count, space):
#we are only called from _drawLines which is one level up
import sys
op = sys._getframe(1).f_locals['op']
raise ValueError(f'Unknown line command {op!a}')
def _drawGrid(self, start, end, weight, color, count, space):
self._drawBox( start, end, weight, color, count, space)
self._drawInnerGrid( start, end, weight, color, count, space)
def _drawBox(self, start, end, weight, color, count, space):
sc,sr = start
ec,er = end
self._drawHLines((sc, sr), (ec, sr), weight, color, count, space)
self._drawHLines((sc, er+1), (ec, er+1), weight, color, count, space)
self._drawVLines((sc, sr), (sc, er), weight, color, count, space)
self._drawVLines((ec+1, sr), (ec+1, er), weight, color, count, space)
def _drawInnerGrid(self, start, end, weight, color, count, space):
sc,sr = start
ec,er = end
self._drawHLines((sc, sr+1), (ec, er), weight, color, count, space)
self._drawVLines((sc+1, sr), (ec, er), weight, color, count, space)
def _prepLine(self, weight, color):
if color and color!=self._curcolor:
self.canv.setStrokeColor(color)
self._curcolor = color
if weight and weight!=self._curweight:
self.canv.setLineWidth(weight)
self._curweight = weight
def _drawHLines(self, start, end, weight, color, count, space):
sc,sr = start
ec,er = end
ecp = self._colpositions[sc:ec+2]
rp = self._rowpositions[sr:er+1]
if len(ecp)<=1 or len(rp)<1: return
self._prepLine(weight, color)
scp = ecp[0]
ecp = ecp[-1]
hBlocks = getattr(self,'_hBlocks',{})
canvLine = self.canv.line
if count == 1:
for y in rp:
_hLine(canvLine, scp, ecp, y, hBlocks)
else:
lf = lambda x0,y0,x1,y1,canvLine=canvLine, ws=weight+space, count=count: _multiLine(x0,x1,y0,canvLine,ws,count)
for y in rp:
_hLine(lf, scp, ecp, y, hBlocks)
def _drawHLinesB(self, start, end, weight, color, count, space):
sc,sr = start
ec,er = end
self._drawHLines((sc, sr+1), (ec, er+1), weight, color, count, space)
def _drawVLines(self, start, end, weight, color, count, space):
sc,sr = start
ec,er = end
erp = self._rowpositions[sr:er+2]
cp = self._colpositions[sc:ec+1]
if len(erp)<=1 or len(cp)<1: return
self._prepLine(weight, color)
srp = erp[0]
erp = erp[-1]
vBlocks = getattr(self,'_vBlocks',{})
canvLine = lambda y0, x0, y1, x1, _line=self.canv.line: _line(x0,y0,x1,y1)
if count == 1:
for x in cp:
_hLine(canvLine, erp, srp, x, vBlocks)
else:
lf = lambda x0,y0,x1,y1,canvLine=canvLine, ws=weight+space, count=count: _multiLine(x0,x1,y0,canvLine,ws,count)
for x in cp:
_hLine(lf, erp, srp, x, vBlocks)
def _drawVLinesA(self, start, end, weight, color, count, space):
sc,sr = start
ec,er = end
self._drawVLines((sc+1, sr), (ec+1, er), weight, color, count, space)
def wrap(self, availWidth, availHeight):
self._calc(availWidth, availHeight)
self.availWidth = availWidth
return (self._width, self._height)
def onSplit(self,T,byRow=1):
'''
This method will be called when the Table is split.
Special purpose tables can override to do special stuff.
'''
pass
def _cr_0(self,n,cmds,nr0,doInRowSplit, _srflMode=False):
#ths is used to modify/apply styles in splitIn row case
#for the first part of a split.
ncols = self._ncols
for c in cmds:
(sc,sr), (ec,er) = c[1:3]
if sr in _SPECIALROWS:
if sr[0]=='i':
self._addCommand(c) #re-append the command
if sr=='inrowsplitstart' and doInRowSplit:
if sc<0: sc+=ncols
if ec<0: ec+=ncols
self._addCommand((c[0],)+((sc, n-1), (ec, n-1))+tuple(c[3:]))
continue
if not _srflMode: continue
self._addCommand(c) #re-append the command
if sr=='splitfirst': continue
sr = er = n-1
if sr<0: sr += nr0
if sr>=n: continue
if er>=n: er = n-1
self._addCommand((c[0],)+((sc, sr), (ec, er))+tuple(c[3:]))
def _cr_1_1(self, n, nRows, repeatRows, cmds, doInRowSplit, _srflMode=False):
nrr = len(repeatRows)
rrS = set(repeatRows)
ncols = self._ncols
for c in cmds:
(sc,sr), (ec,er) = c[1:3]
if sr in _SPECIALROWS:
if sr[0]=='i':
self._addCommand(c) #re-append the command
if sr=='inrowsplitend' and doInRowSplit:
if sc<0: sc+=ncols
if ec<0: ec+=ncols
self._addCommand((c[0],)+((sc, nrr), (ec, nrr))+tuple(c[3:]))
continue
if not _srflMode: continue
self._addCommand(c)
if sr=='splitlast': continue
sr = er = n
if sr<0: sr += nRows
if er<0: er += nRows
cS = set(range(sr,er+1)) & rrS
if cS:
#it's a repeat row
cS = list(cS)
self._addCommand((c[0],)+((sc, repeatRows.index(min(cS))), (ec, repeatRows.index(max(cS))))+tuple(c[3:]))
if er<n: continue
sr = max(sr-n,0)+nrr
er = max(er-n,0)+nrr
self._addCommand((c[0],)+((sc, sr), (ec, er))+tuple(c[3:]))
sr = self._rowSplitRange
if sr:
sr, er = sr
if sr<0: sr += nRows
if er<0: er += nRows
if er<n:
self._rowSplitRange = None
else:
sr = max(sr-n,0)+nrr
er = max(er-n,0)+nrr
self._rowSplitRange = sr,er
def _cr_1_0(self,n,cmds,doInRowSplit,_srflMode=False):
for c in cmds:
(sc,sr), (ec,er) = c[1:3]
if sr in _SPECIALROWS:
if sr[0]=='i':
self._addCommand(c) #re-append the command
if sr=='inrowsplitend' and doInRowSplit:
if sc<0: sc+=ncols
if ec<0: ec+=ncols
self._addCommand((c[0],)+((sc, 0), (ec, 0))+tuple(c[3:]))
continue
if not _srflMode: continue
self._addCommand(c)
if sr=='splitlast': continue
sr = er = n
if er>=0 and er<n: continue
if sr>=0 and sr<n: sr=0
if sr>=n: sr -= n
if er>=n: er -= n
self._addCommand((c[0],)+((sc, sr), (ec, er))+tuple(c[3:]))
def _splitCell(self, value, style, oldHeight, newHeight, width):
# Content height of the new top row
height0 = newHeight - style.topPadding
# Content height of the new bottom row
height1 = oldHeight - (style.topPadding + newHeight)
if isinstance(value, (tuple, list)):
newCellContent = []
postponedContent = []
split = False
cellHeight = self._listCellGeom(value, width, style)[1]
if style.valign == "MIDDLE":
usedHeight = (oldHeight - cellHeight) / 2
else:
usedHeight = 0
for flowable in value:
if split:
if flowable.height <= height1:
postponedContent.append(flowable)
# Shrink the available height:
height1 -= flowable.height
else:
# The content doesn't fit after the split:
return []
elif usedHeight + flowable.height <= height0:
newCellContent.append(flowable)
usedHeight += flowable.height
else:
# This is where we need to split
splits = flowable.split(width, height0-usedHeight)
if splits:
newCellContent.append(splits[0])
postponedContent.append(splits[1])
else:
# We couldn't split this flowable at the desired
# point. If we already has added previous paragraphs
# to the content, just add everything after the split.
# Also try adding it after the split if valign isn't TOP
if newCellContent or style.valign != "TOP":
if flowable.height <= height1:
postponedContent.append(flowable)
# Shrink the available height:
height1 -= flowable.height
else:
# The content doesn't fit after the split:
return []
else:
# We could not split this, so we fail:
return []
split = True
return (tuple(newCellContent), tuple(postponedContent))
elif isinstance(value, str):
rows = value.split("\n")
lineHeight = 1.2 * style.fontsize
contentHeight = (style.leading or lineHeight) * len(rows)
if style.valign == "TOP" and contentHeight <= height0:
# This fits in the first cell, all is good
return (value, '')
elif style.valign == "BOTTOM" and contentHeight <= height1:
# This fits in the second cell, all is good
return ('', value)
elif style.valign == "MIDDLE":
# Put it in the largest cell:
if height1 > height0:
return ('', value)
else:
return (value, '')
elif len(rows) < 2:
# It doesn't fit, and there's nothing to split: Fail
return []
# We need to split this, and there are multiple lines, so we can
if style.valign == "TOP":
splitPoint = height0 // lineHeight
elif style.valign == "BOTTOM":
splitPoint = len(rows) - (height1 // lineHeight)
else: # MID
splitPoint = (height0 - height1 + contentHeight) // (2 * lineHeight)
splitPoint = int(splitPoint)
return ('\n'.join(rows[:splitPoint]), '\n'.join(rows[splitPoint:]))
# No content
return ('', '')
def _splitLineCmds(self, n, doInRowSplit=0):
nrows = self._nrows
ncols = self._ncols
#copy the commands
A = []
# hack up the line commands
for op, (sc,sr), (ec,er), weight, color, cap, dash, join, count, space in self._linecmds:
if isinstance(sr,strTypes) and sr in _SPECIALROWS:
A.append((op,(sc,sr), (ec,sr), weight, color, cap, dash, join, count, space))
if sr=='splitlast':
sr = er = n-1
elif sr=='splitfirst':
sr = n
er = n
else:
if sc < 0: sc += ncols
if ec < 0: ec += ncols
A[-1] = (op,(sc,sr), (ec,sr), weight, color, cap, dash, join, count, space)
continue
if sc < 0: sc += ncols
if ec < 0: ec += ncols
if sr < 0: sr += nrows
if er < 0: er += nrows
if op in ('BOX','OUTLINE','GRID'):
if (sr<n and er>=n) or (doInRowSplit and sr==n):
# we have to split the BOX
A.append(('LINEABOVE',(sc,sr), (ec,sr), weight, color, cap, dash, join, count, space))
A.append(('LINEBEFORE',(sc,sr), (sc,er), weight, color, cap, dash, join, count, space))
A.append(('LINEAFTER',(ec,sr), (ec,er), weight, color, cap, dash, join, count, space))
A.append(('LINEBELOW',(sc,er), (ec,er), weight, color, cap, dash, join, count, space))
if op=='GRID':
if doInRowSplit:
A.append(('INNERGRID',(sc,sr), (ec,n-1), weight, color, cap, dash, join, count, space))
A.append(('INNERGRID',(sc,n), (ec,er), weight, color, cap, dash, join, count, space))
else:
A.append(('LINEBELOW',(sc,n-1), (ec,n-1), weight, color, cap, dash, join, count, space))
A.append(('LINEABOVE',(sc,n), (ec,n), weight, color, cap, dash, join, count, space))
A.append(('INNERGRID',(sc,sr), (ec,er), weight, color, cap, dash, join, count, space))
else:
A.append((op,(sc,sr), (ec,er), weight, color, cap, dash, join, count, space))
elif op == 'INNERGRID':
if sr<n and er>=n and not doInRowSplit:
A.append(('LINEBELOW',(sc,n-1), (ec,n-1), weight, color, cap, dash, join, count, space))
A.append(('LINEABOVE',(sc,n), (ec,n), weight, color, cap, dash, join, count, space))
A.append((op,(sc,sr), (ec,er), weight, color, cap, dash, join, count, space))
elif op == 'LINEBELOW':
if sr<n and er>=(n-1):
A.append(('LINEABOVE',(sc,n), (ec,n), weight, color, cap, dash, join, count, space))
A.append((op,(sc,sr), (ec,er), weight, color, cap, dash, join, count, space))
elif op == 'LINEABOVE':
if sr<=n and er>=n:
A.append(('LINEBELOW',(sc,n-1), (ec,n-1), weight, color, cap, dash, join, count, space))
A.append((op,(sc,sr), (ec,er), weight, color, cap, dash, join, count, space))
else:
A.append((op,(sc,sr), (ec,er), weight, color, cap, dash, join, count, space))
return A
def _stretchCommands(self, n, cmds, oldrowcount):
"""Stretches the commands when a row is split
The row start is sr, the row end is er.
sr | er | result
---------------------------------------------------------------------
<n | <n | Do nothing.
| >=n | A command that spans the break, extend end.
---------------------------------------------------------------------
==n | ==n | Zero height. Extend the end, unless it's a LINEABOVE
| | commands, it's between rows so do nothing.
| | For LINEBELOW increase both.
| >n | A command that spans the break, extend end.
---------------------------------------------------------------------
>n | >n | This command comes after the break, increase both.
---------------------------------------------------------------------
Summary:
1. If er > n then increase er
2. If sr > n then increase sr
3. If er == n and sr < n, increase er
4. If er == sr == n and cmd is not line, increase er
"""
stretched = [].append
for c in cmds:
cmd, (sc,sr), (ec,er) = c[0:3]
if sr in _SPECIALROWS or er in _SPECIALROWS:
stretched(c)
continue
if er < 0:
er += oldrowcount
if sr < 0:
sr += oldrowcount
if er > n:
er += 1
elif er == n:
if sr < n or (sr == n and cmd != "LINEABOVE"):
er += 1
if sr > n or (sr == n and cmd == "LINEBELOW"):
sr += 1
stretched((cmd, (sc,sr), (ec,er)) + c[3:])
return stretched.__self__
def _splitRows(self,availHeight,doInRowSplit=0):
# Get the split position. if we split between rows (doInRowSplit=0),
# then n will be the first row after the split. If we split a row,
# then n is the row we split in two.
n=self._getFirstPossibleSplitRowPosition(availHeight,ignoreSpans=doInRowSplit)
# We can't split before or in the repeatRows/headers
repeatRows = self.repeatRows
maxrepeat = repeatRows if isinstance(repeatRows,int) else max(repeatRows)+1
if doInRowSplit and n<maxrepeat or not doInRowSplit and n<=maxrepeat:
return []
# If the whole table fits, return it
lim = len(self._rowHeights)
if n==lim: return [self]
lo = self._rowSplitRange
if lo:
lo, hi = lo
if lo<0: lo += lim
if hi<0: hi += lim
if n>hi:
return self._splitRows(availHeight - sum(self._rowHeights[hi:n]), doInRowSplit=doInRowSplit)
elif n<lo:
return []
repeatCols = self.repeatCols
if not doInRowSplit:
T = self
data = self._cellvalues
else:
data = [_[:] for _ in self._cellvalues]
# We are splitting the n row into two, if possible.
# We can't split if the available height is less than the minimum set:
if self._minRowHeights and availHeight < self._minRowHeights[n]:
return []
usedHeights = sum(self._rowHeights[:n])
cellvalues = self._cellvalues[n]
cellStyles = self._cellStyles[n]
cellWidths = self._colWidths
curRowHeight = self._rowHeights[n]
# First find the min/max split point
minSplit = 0 # Counted from top
maxSplit = 0 # Counted from bottom
maxHeight = 0
for column, (value, style, width) in enumerate(zip(cellvalues, cellStyles, cellWidths)):
if self._spanCmds and self._spanRanges.get((column, n), None) is None:
# This is part of another cell span, the value will not be displayed
continue
if isinstance(value, (tuple, list)):
# A sequence of flowables:
w, height = self._listCellGeom(value, width, style)
height += style.topPadding + style.bottomPadding
if height > maxHeight:
maxHeight = height
elif isinstance(value, str):
rows = value.split("\n")
lineHeight = 1.2 * style.fontsize
height = lineHeight * len(rows) + style.topPadding + style.bottomPadding
# Make sure we don't try to split in the middle of the first or last line
minSplit = max(minSplit, lineHeight + style.topPadding)
maxSplit = max(maxSplit, lineHeight + style.bottomPadding)
if height > maxHeight:
maxHeight = height
if ((minSplit + maxSplit > curRowHeight) or
(minSplit > (availHeight - usedHeights))):
# We can't split this row. So we should fail, and let the split get retried
# with splitInRow = 0. However, if there is a spanned row that will also
# fail. So first we need to check if any cell in the current row is spanned.
if not self._spanCmds:
# There are no spans to look at, so we can skip this and fail directly
return []
splitCells = set()
for column in range(self._ncols):
cell = (column, n)
if (cell in self._rowSpanCells and
self._spanRanges.get((column, n), None) is None):
# This cell is a part of a rowSpan, and not the main cell.
# Find the real cell and cell value
for cell, span in self._spanRanges.items():
if span is None:
continue
start_col, start_row, end_col, end_row = span
if (column >= start_col and
column <= end_col and
n > start_row and
n <= end_row):
splitCells.add(cell)
break
if not splitCells:
# There were no spanned rows that could be split, so we fail.
return []
spanCmds = []
for cmd, (sc, sr), (ec, er) in self._spanCmds:
# -1 means last row/column, handle that here.
if sc < 0:
sc += self._ncols
if ec < 0:
ec += self._ncols
if sr < 0:
sr += self._nrows
if er < 0:
er += self._nrows
spanCmds.append((cmd, (sc, sr), (ec, er)))
newCellStyles = [_[:] for _ in self._cellStyles]
bkgrndcmds = self._bkgrndcmds
# There are cells spanning the rows we want to split. They can be split,
# because the _getFirstPossibleSplitRowPosition() call above checked
# that they are not in a nosplit range, so let's split them.
for cell in splitCells:
span_sc, span_sr, span_ec, span_er = self._spanRanges[cell]
spanRect = self._spanRects[cell]
oldHeight = spanRect[3]
newHeight = sum(self._rowHeights[span_sr:n])
# Copy the style:
oldStyle = newCellStyles[span_sr][span_sc]
res = self._splitCell(self._cellvalues[span_sr][span_sc],
oldStyle, oldHeight, newHeight, width)
if not res:
# Could not split
return []
# Replace the data values:
data[span_sr][span_sc] = res[0]
data[n][span_sc] = res[1]
# Now get replace the rowspan with two new rowspans, or
# remove the spans if no span remains
newSpanCmds = []
for cmd, start, end in spanCmds:
if ((span_sc, span_sr) == start and
(span_ec, span_er) == end):
# Modify this:
if n-1 > span_sr or span_sc != span_ec:
newSpanCmds.append((cmd, (span_sc, span_sr), (span_ec, n-1)))
if n < span_er or span_sc != span_ec:
newSpanCmds.append((cmd, (span_sc, n), (span_ec, span_er)))
else:
newSpanCmds.append((cmd, start, end))
spanCmds = newSpanCmds
newbkgrndcmds = []
for cmd, start, end, color in bkgrndcmds:
if start == (span_sc, span_sr):
# The cell we are splitting has a background color command.
# Add commands for the new split cells:
newbkgrndcmds.append((cmd, start, (end[0], n-1), color))
newbkgrndcmds.append((cmd, (start[0], n), (end[0], n), color))
else:
newbkgrndcmds.append((cmd, start, end, color))
bkgrndcmds = newbkgrndcmds
# And adjust the style
newStyle = oldStyle.copy()
if oldStyle.valign == "MIDDLE":
# Adjust margins
if res[0] and res[1]:
# We split the content, so fix up the valign:
oldStyle.valign = "BOTTOM"
newStyle.valign = "TOP"
else:
# Adjust the margins to push it towards the true middle
h = self._listCellGeom(v[0] or v[1], width, oldStyle)[1]
margin = (curRowHeight - h) / 2
if v[0]:
oldStyle.topPadding += margin
elif v[1]:
newStyle.bottomPadding += margin
newCellStyles[n][span_sc] = newStyle
# Make a new table here
T = self.__class__( data, colWidths=self._colWidths,
rowHeights=self._rowHeights, repeatRows=self.repeatRows,
repeatCols=self.repeatCols, splitByRow=self.splitByRow,
splitInRow=self.splitInRow, normalizedData=1,
cellStyles=newCellStyles, ident=self.ident,
spaceBefore=getattr(self,'spaceBefore',None),
longTableOptimize=self._longTableOptimize,
cornerRadii=getattr(self,'_cornerRadii',None),
renderCB=getattr(self,'_renderCB',None),
)
T._bkgrndcmds = bkgrndcmds
T._spanCmds = spanCmds
T._nosplitCmds = self._nosplitCmds
T._srflcmds = self._srflcmds
T._sircmds = self._sircmds
T._colpositions = self._colpositions
T._rowpositions = self._rowpositions
T._calcNoSplitRanges()
T._calcSpanRanges()
T._calcSpanRects()
# And then, remove any lines that now appear that were inside
# the spanning cell before the split. First, we need to split
# the grids, and convert the bit of the grid that spans the
# split to a line.
newlinecmds = []
for linecmd in self._linecmds:
op, (sc,sr), (ec,er), weight, color, cap, dash, join, count, space = linecmd
# -1 means "to the end", so we handle that here
if er < 0:
er += T._nrows
if ec < 0:
ec += T._ncols
if ((op == 'BOX') or (op == 'GRID' and (sr <= n or er >=n)) or
(op == 'INNERGRID' and (sr < n or er > n))):
if op in ('GRID', 'INNERGRID'):
newlinecmds.append(('INNERGRID',(sc,sr), (ec,n-1), weight, color, cap, dash, join, count, space))
newlinecmds.append(('INNERGRID',(sc,n), (ec,er), weight, color, cap, dash, join, count, space))
newlinecmds.append(('LINEBELOW',(sc,n-1), (ec,n-1), weight, color, cap, dash, join, count, space))
if op in ('GRID', 'BOX'):
# The box must be made of lines, because otherwise
# it might not get split where it should, below.
newlinecmds.append(('LINEABOVE', (sc,sr), (ec,sr), weight, color, cap, dash, join, count, space))
newlinecmds.append(('LINEBELOW', (sc,er), (ec,er), weight, color, cap, dash, join, count, space))
newlinecmds.append(('LINEBEFORE', (sc,sr), (sc,er), weight, color, cap, dash, join, count, space))
newlinecmds.append(('LINEAFTER', (ec,sr), (ec,er), weight, color, cap, dash, join, count, space))
else:
newlinecmds.append(linecmd)
continue
# Then secondly split any LINEABOVE and LINEBELOW so that the
# split cells don't get them.
for cell in splitCells:
moddedcmds = []
for linecmd in newlinecmds:
op, (sc,sr), (ec,er), weight, color, cap, dash, join, count, space = linecmd
span_sc, span_sr, span_ec, span_er = self._spanRanges[cell]
if (((op == "LINEABOVE" and er > span_sr and sr <= span_er) or
(op == "LINEBELOW" and er >= span_sr and sr < span_er)) and
(sc <= span_ec and ec >= span_sc)):
# This needs handling of some sort
if op == "LINEABOVE":
startrow = span_sr
endrow = span_er + 1
else:
startrow = span_sr - 1
endrow = span_er
if sr <= startrow:
# Anything before the span should be unaffected:
moddedcmds.append(
(op, (sc, sr), (ec, startrow), weight, color, cap, dash, join, count, space)
)
# For any lines in between we need to remove the split cell
if span_sc > sc:
# The start column of the span is higher than the start column of
# the line. So we need a line up until but not including the start column
moddedcmds.append(
(op, (sc, max(startrow, sr)), (span_sc-1, min(er, endrow)), weight, color, cap, dash, join, count, space)
)
if span_ec < ec:
# The start column of the span is lower than the end column of
# the line. So we need a line up starting after but not including the end column
moddedcmds.append(
(op, (span_ec+1, max(startrow, sr)), (ec, min(er, endrow)), weight, color, cap, dash, join, count, space)
)
if er >= endrow:
moddedcmds.append(
(op, (sc, endrow), (ec, er), weight, color, cap, dash, join, count, space)
)
# Anything after the span should be unaffected:
else:
moddedcmds.append(linecmd)
newlinecmds = moddedcmds
T._linecmds = newlinecmds
return T._splitRows(availHeight,doInRowSplit=False)
# This is where we split the row:
splitPoint = min(availHeight - usedHeights, maxHeight - maxSplit)
if splitPoint+1 < self.splitInRow:
# The height of the split is smaller than the minimum.
# Fail, and the whole table will be moved to the next page.
return []
remaining = self._height - splitPoint
if remaining < self.splitInRow:
# The remaining height of the table is smaller than the minimum.
# Fail, and the whole table will be moved to the next page.
return []
R0 = [] # Top half of the row
R0Height = 0 # Minimum height
R1 = [] # Bottom half of the row
R1Height = 0 # Minimum height
R1Styles = []
for (value, style, width) in zip(cellvalues, cellStyles, cellWidths):
v = self._splitCell(value, style, curRowHeight, splitPoint, width)
if not v:
# Splitting the table failed
return []
newStyle = style.copy()
if style.valign == "MIDDLE":
# Adjust margins
if v[0] and v[1]:
# We split the content, so fix up the valign:
style.valign = "BOTTOM"
newStyle.valign = "TOP"
else:
# Adjust the margins to push it towards the true middle
h = self._listCellGeom(v[0] or v[1], width, style)[1]
margin = (curRowHeight - h) / 2
if v[0]:
style.topPadding += margin
elif v[1]:
newStyle.bottomPadding += margin
R0.append(v[0])
R1.append(v[1])
h0 = self._listCellGeom(v[0], width, style)[1] + style.topPadding + style.bottomPadding
R0Height = max(R0Height, h0)
h1 = self._listCellGeom(v[1], width, style)[1] + style.topPadding + style.bottomPadding
R1Height = max(R1Height, h1)
R1Styles.append(newStyle)
# Make a new table with the row split into two:
usedHeight = min(splitPoint, R0Height)
newRowHeight = max(R1Height, self._rowHeights[n] - usedHeight)
newRowHeights = self._rowHeights[:]
newRowHeights.insert(n + 1, newRowHeight)
newRowHeights[n] = usedHeight
newCellStyles = self._cellStyles[:]
newCellStyles.insert(n + 1, R1Styles)
data = data[:n] + [R0] + [R1] + data[n+1:]
T = self.__class__( data, colWidths=self._colWidths,
rowHeights=newRowHeights, repeatRows=self.repeatRows,
repeatCols=self.repeatCols, splitByRow=self.splitByRow,
splitInRow=self.splitInRow, normalizedData=1,
cellStyles=newCellStyles, ident=self.ident,
spaceBefore=getattr(self,'spaceBefore',None),
longTableOptimize=self._longTableOptimize,
cornerRadii=getattr(self,'_cornerRadii',None),
renderCB=getattr(self,'_renderCB',None),
)
T._linecmds = self._stretchCommands(n, self._linecmds, lim)
T._bkgrndcmds = self._stretchCommands(n, self._bkgrndcmds, lim)
T._spanCmds = self._stretchCommands(n, self._spanCmds, lim)
T._nosplitCmds = self._stretchCommands(n, self._nosplitCmds, lim)
T._srflcmds = self._stretchCommands(n, self._srflcmds, lim)
T._sircmds = self._stretchCommands(n, self._sircmds, lim)
n = n + 1
#we're going to split into two superRows
ident = self.ident
if ident: ident = IdentStr(ident)
lto = T._longTableOptimize
if lto:
splitH = T._rowHeights
else:
splitH = T._argH
cornerRadii = getattr(self,'_cornerRadii',None)
renderCB = getattr(self,'_renderCB',None)
R0 = self.__class__( data[:n], colWidths=T._colWidths, rowHeights=splitH[:n],
repeatRows=repeatRows, repeatCols=repeatCols, splitByRow=self.splitByRow,
splitInRow=self.splitInRow, normalizedData=1, cellStyles=T._cellStyles[:n],
ident=ident,
spaceBefore=getattr(self,'spaceBefore',None),
longTableOptimize=lto,
cornerRadii=cornerRadii[:2] if cornerRadii else None,
renderCB=renderCB,
)
nrows = T._nrows
ncols = T._ncols
_linecmds = T._splitLineCmds(n, doInRowSplit=doInRowSplit)
R0._cr_0(n,_linecmds,nrows,doInRowSplit)
R0._cr_0(n,T._bkgrndcmds,nrows,doInRowSplit,_srflMode=True)
R0._cr_0(n,T._spanCmds,nrows,doInRowSplit)
R0._cr_0(n,T._nosplitCmds,nrows,doInRowSplit)
for c in T._srflcmds:
R0._addCommand(c)
if c[1][1]!='splitlast': continue
(sc,sr), (ec,er) = c[1:3]
R0._addCommand((c[0],)+((sc, n-1), (ec, n-1))+tuple(c[3:]))
if ident: ident = IdentStr(ident)
if repeatRows:
if isinstance(repeatRows,int):
iRows = data[:repeatRows]
iRowH = splitH[:repeatRows]
iCS = T._cellStyles[:repeatRows]
repeatRows = list(range(repeatRows))
else:
#we have a list of repeated rows eg (1,3)
repeatRows = list(sorted(repeatRows))
iRows = [data[i] for i in repeatRows]
iRowH = [splitH[i] for i in repeatRows]
iCS = [T._cellStyles[i] for i in repeatRows]
R1 = self.__class__(iRows+data[n:],colWidths=T._colWidths,
rowHeights=iRowH+splitH[n:],
repeatRows=len(repeatRows), repeatCols=repeatCols,
splitByRow=self.splitByRow, splitInRow=self.splitInRow,
normalizedData=1,
cellStyles=iCS+T._cellStyles[n:],
ident=ident,
spaceAfter=getattr(self,'spaceAfter',None),
longTableOptimize=lto,
cornerRadii = cornerRadii,
renderCB = renderCB,
)
R1._cr_1_1(n,nrows,repeatRows,_linecmds,doInRowSplit)
R1._cr_1_1(n,nrows,repeatRows,T._bkgrndcmds,doInRowSplit,_srflMode=True)
R1._cr_1_1(n,nrows,repeatRows,T._spanCmds,doInRowSplit)
R1._cr_1_1(n,nrows,repeatRows,T._nosplitCmds,doInRowSplit)
else:
#R1 = slelf.__class__(data[n:], self._argW, self._argH[n:],
R1 = self.__class__(data[n:], colWidths=T._colWidths, rowHeights=splitH[n:],
repeatRows=repeatRows, repeatCols=repeatCols,
splitByRow=self.splitByRow, splitInRow=self.splitInRow,
normalizedData=1, cellStyles=T._cellStyles[n:],
ident=ident,
spaceAfter=getattr(self,'spaceAfter',None),
longTableOptimize=lto,
cornerRadii = ([0,0] + cornerRadii[2:]) if cornerRadii else None,
renderCB = renderCB,
)
R1._cr_1_0(n,_linecmds,doInRowSplit)
R1._cr_1_0(n,T._bkgrndcmds,doInRowSplit,_srflMode=True)
R1._cr_1_0(n,T._spanCmds,doInRowSplit)
R1._cr_1_0(n,T._nosplitCmds,doInRowSplit)
for c in T._srflcmds:
R1._addCommand(c)
if c[1][1]!='splitfirst': continue
(sc,sr), (ec,er) = c[1:3]
R1._addCommand((c[0],)+((sc, 0), (ec, 0))+tuple(c[3:]))
R0.hAlign = R1.hAlign = T.hAlign
R0.vAlign = R1.vAlign = T.vAlign
self.onSplit(R0)
self.onSplit(R1)
return [R0,R1]
def _getRowImpossible(impossible,cells,ranges):
for xy in cells:
r=ranges[xy]
if r!=None:
y1,y2=r[1],r[3]
if y1!=y2:
ymin=min(y1,y2) #normalize
ymax=max(y1,y2) #normalize
y=ymin+1
while 1:
if y>ymax: break
impossible[y]=None #split at position y is impossible because of overlapping rowspan
y+=1
_getRowImpossible=staticmethod(_getRowImpossible)
def _getFirstPossibleSplitRowPosition(self,availHeight,ignoreSpans=0):
# Returns the LAST possible split row position
impossible={}
# With inRowSplits we ignore spans
if self._spanCmds and not ignoreSpans:
self._getRowImpossible(impossible,self._rowSpanCells,self._spanRanges)
if self._nosplitCmds:
self._getRowImpossible(impossible,self._rowNoSplitCells,self._nosplitRanges)
h = 0
n = 1
split_at = 0 # from this point of view 0 is the first position where the table may *always* be splitted
for rh in self._rowHeights:
if h+rh>availHeight:
break
if n not in impossible:
split_at=n
h=h+rh
n=n+1
return split_at
def split(self, availWidth, availHeight):
self._calc(availWidth, availHeight)
if self.splitByRow or self.splitInRow:
if not rl_config.allowTableBoundsErrors and self._width>availWidth: return []
# If self.splitByRow is true, first try with doInRowSplit as false.
# Otherwise, first try with doInRowSplit as true
result = self._splitRows(availHeight, doInRowSplit=not self.splitByRow)
if result:
# That worked, return that:
return result
# The first attempt did NOT succeed, now try with the flag flipped
# (unless self.splitInRow is false)
if self.splitInRow:
return self._splitRows(availHeight, doInRowSplit=self.splitByRow)
# We can't split this table in any way, raise an error:
return []
def _makeRoundedCornersClip(self, FUZZ=rl_config._FUZZ):
self._roundingRectDef = None
cornerRadii = getattr(self,'_cornerRadii',None)
if not cornerRadii or max(cornerRadii)<=FUZZ: return
nrows = self._nrows
ncols = self._ncols
ar = [min(self._rowHeights[i],self._colWidths[j],cornerRadii[k]) for
k,(i,j) in enumerate((
(0,0),
(0,ncols-1),
(nrows-1,0),
(nrows-1, ncols-1),
))]
rp = self._rowpositions
cp = self._colpositions
x0 = cp[0]
y0 = rp[nrows]
x1 = cp[ncols]
y1 = rp[0]
w = x1 - x0
h = y1 - y0
self._roundingRectDef = RoundingRectDef(x0, y0, w, h, x1, y1, ar, [])
P = self.canv.beginPath()
P.roundRect(x0, y0, w, h, ar)
c = self.canv
c.addLiteral('%begin table rect clip')
c.clipPath(P,stroke=0)
c.addLiteral('%end table rect clip')
def _restoreRoundingObscuredLines(self):
x0, y0, w, h, x1, y1, ar, SL = self._roundingRectDef
if not SL: return
canv = self.canv
canv.saveState()
ccap = cdash = cjoin = self._curweight = self._curcolor = None
line = canv.line
cornerRadii = self._cornerRadii
for (xs,ys,xe,ye,weight,color,cap,dash,join) in SL:
if cap!=None and ccap!=cap:
canv.setLineCap(cap)
ccap = cap
if dash is None or dash == []:
if cdash is not None:
canv.setDash()
cdash = None
elif dash != cdash:
canv.setDash(dash)
cdash = dash
if join is not None and cjoin!=join:
canv.setLineJoin(join)
cjoin = join
self._prepLine(weight, color)
if ys==ye:
#horizontal line
if ys>y1 or ys<y0:
line(xs,ys,xe,ye) #simple line that's outside the clip
continue
#which corners are involved
if ys==y0:
ypos = 'bottom'
r0 = ar[2]
r1 = ar[3]
else: #ys==y1
ypos = 'top'
r0 = ar[0]
r1 = ar[1]
if xs>=x0+r0 and xe<=x1-r1:
line(xs,ys,xe,ye) #simple line with no rounding
continue
#we have some rounding so we must use a path
c0 = _quadrantDef('left',ypos,(xs,ys), r0, kind=2, direction='left-right') if xs<x0+r0 else None
c1 = _quadrantDef('right',ypos,(xe,ye), r1, kind=1, direction='left-right') if xe>x1-r1 else None
else:
#vertical line
if xs>x1 or xs<x0:
line(xs,ys,xe,ye) #simple line that's outside the clip
continue
#which corners are involved
if xs==x0:
xpos = 'left'
r0 = ar[2]
r1 = ar[0]
else: #xs==x1
xpos = 'right'
r0 = ar[3]
r1 = ar[1]
if ys>=y0+r0 and ye<=y1-r1:
line(xs,ys,xe,ye) #simple line with no rounding
continue
#we have some rounding so we must use a path
c0 = _quadrantDef(xpos,'bottom',(xs,ys), r0, kind=2, direction='bottom-top') if ys<y0+r0 else None
c1 = _quadrantDef(xpos,'top',(xe,ye), r1, kind=1, direction='bottom-top') if ye>y1-r1 else None
P = canv.beginPath()
if c0:
P.moveTo(*c0[0])
P.curveTo(c0[1][0],c0[1][1],c0[2][0],c0[2][1], c0[3][0],c0[3][1])
else:
P.moveTo(xs,ys)
if not c1:
P.lineTo(xe,ye)
else:
P.lineTo(*c1[0])
P.curveTo(c1[1][0],c1[1][1],c1[2][0],c1[2][1], c1[3][0],c1[3][1])
canv.drawPath(P, stroke=1, fill=0)
canv.restoreState()
def draw(self):
c = self.canv
c.saveState()
self._curweight = self._curcolor = self._curcellstyle = None
renderCB = getattr(self,'_renderCB')
if renderCB:
renderCB(self,'startTable')
renderCB(self,'startBG')
self._makeRoundedCornersClip()
self._drawBkgrnd()
if renderCB: renderCB(self,'endBG')
if not self._spanCmds:
# old fashioned case, no spanning, steam on and do each cell
if renderCB:
for rowNo, (row, rowstyle, rowpos, rowheight) in enumerate(zip(self._cellvalues, self._cellStyles, self._rowpositions[1:], self._rowHeights)):
renderCB(self,'startRow',rowNo)
for colNo, (cellval, cellstyle, colpos, colwidth) in enumerate(zip(row, rowstyle, self._colpositions[:-1], self._colWidths)):
renderCB(self,'startCell', rowNo, colNo, cellval, cellstyle, (colpos, rowpos), (colwidth, rowheight))
self._drawCell(cellval, cellstyle, (colpos, rowpos), (colwidth, rowheight))
renderCB(self,'endCell')
renderCB(self,'endRow')
else:
for row, rowstyle, rowpos, rowheight in zip(self._cellvalues, self._cellStyles, self._rowpositions[1:], self._rowHeights):
for colNo, (cellval, cellstyle, colpos, colwidth) in enumerate(zip(row, rowstyle, self._colpositions[:-1], self._colWidths)):
self._drawCell(cellval, cellstyle, (colpos, rowpos), (colwidth, rowheight))
else:
# we have some row or col spans, need a more complex algorithm
# to find the rect for each
if renderCB:
for rowNo in range(self._nrows):
renderCB(self,'startRow',rowNo)
for colNo in range(self._ncols):
cellRect = self._spanRects[colNo, rowNo]
if cellRect is not None:
(x, y, width, height) = cellRect
cellval = self._cellvalues[rowNo][colNo]
cellstyle = self._cellStyles[rowNo][colNo]
renderCB(self,'startCell', rowNo, colNo, cellval, cellstyle, (x, y), (width, height))
self._drawCell(cellval, cellstyle, (x, y), (width, height))
renderCB(self,'endCell')
renderCB(self,'endRow')
else:
for rowNo in range(self._nrows):
for colNo in range(self._ncols):
cellRect = self._spanRects[colNo, rowNo]
if cellRect is not None:
(x, y, width, height) = cellRect
cellval = self._cellvalues[rowNo][colNo]
cellstyle = self._cellStyles[rowNo][colNo]
self._drawCell(cellval, cellstyle, (x, y), (width, height))
if renderCB: renderCB(self,'startLines')
self._drawLines()
if renderCB:
renderCB(self,'endLines')
renderCB(self,'endTable')
c.restoreState()
if self._roundingRectDef:
self._restoreRoundingObscuredLines()
def _drawBkgrnd(self):
nrows = self._nrows
ncols = self._ncols
canv = self.canv
colpositions = self._colpositions
rowpositions = self._rowpositions
rowHeights = self._rowHeights
colWidths = self._colWidths
spanRects = getattr(self,'_spanRects',None)
for cmd, (sc, sr), (ec, er), arg in self._bkgrndcmds:
if sr in _SPECIALROWS: continue
if sc < 0: sc = sc + ncols
if ec < 0: ec = ec + ncols
if sr < 0: sr = sr + nrows
if er < 0: er = er + nrows
x0 = colpositions[sc]
y0 = rowpositions[sr]
x1 = colpositions[min(ec+1,ncols)]
y1 = rowpositions[min(er+1,nrows)]
w, h = x1-x0, y1-y0
if hasattr(arg,'__call__'):
arg(self,canv, x0, y0, w, h)
elif cmd == 'ROWBACKGROUNDS':
#Need a list of colors to cycle through. The arguments
#might be already colours, or convertible to colors, or
# None, or the str 'None'.
#It's very common to alternate a pale shade with None.
colorCycle = list(map(colors.toColorOrNone, arg))
count = len(colorCycle)
rowCount = er - sr + 1
for i in range(rowCount):
color = colorCycle[i%count]
h = rowHeights[sr + i]
if color:
canv.setFillColor(color)
canv.rect(x0, y0, w, -h, stroke=0,fill=1)
y0 = y0 - h
elif cmd == 'COLBACKGROUNDS':
#cycle through colours columnwise
colorCycle = list(map(colors.toColorOrNone, arg))
count = len(colorCycle)
colCount = ec - sc + 1
for i in range(colCount):
color = colorCycle[i%count]
w = colWidths[sc + i]
if color:
canv.setFillColor(color)
canv.rect(x0, y0, w, h, stroke=0,fill=1)
x0 = x0 +w
else: #cmd=='BACKGROUND'
if (arg and isinstance(arg,(list,tuple))
and arg[0] in ('VERTICAL','HORIZONTAL', 'VERTICAL2', 'HORIZONTAL2',
'LINEARGRADIENT', 'RADIALGRADIENT')):
if ec==sc and er==sr and spanRects:
xywh = spanRects.get((sc,sr))
if xywh:
#it's a single spanned cell
x0, y0, w, h = xywh
arg0, arg = arg[0], arg[1:]
#
# arg is a list, assume we are going for a gradient fill
# where we expect a containing a direction for the gradient
# and the starting an final gradient colors. For example:
# ['HORIZONTAL', colors.white, colors.grey] or
# ['VERTICAL', colors.red, colors.blue]
#
canv.saveState()
p = canv.beginPath()
p.rect(x0, y0, w, h)
canv.clipPath(p, stroke=0)
if arg0=="HORIZONTAL":
canv.linearGradient(x0,y0,x0+w,y0,arg,extend=False)
elif arg0 == "HORIZONTAL2":
xh = x0 + w/2.0
canv.linearGradient(x0, y0, xh, y0, arg, extend=False)
canv.linearGradient(xh, y0, x0 + w, y0, arg[::-1], extend=False)
#canv.linearGradient(x0, y0, x0 + w, y0, arg+arg[1::-1], extend=False)
elif arg0 == "VERTICAL2":
yh = y0 + h/2.0
canv.linearGradient(x0, y0, x0, yh, arg, extend=False)
canv.linearGradient(x0, yh, x0, y0 + h, arg[::-1], extend=False)
#canv.linearGradient(x0, y0, x0, y0 + h, arg+arg[1::-1], extend=False)
elif arg0=="VERTICAL":
canv.linearGradient(x0, y0, x0, y0 + h, arg, extend=False)
elif arg0=='LINEARGRADIENT':
# the remaining arguments define the axis, extend, colors, stops as
# axis = (x0,y0, x1, y1) given as fractions of width / height
# extend = bool or [bool, bool]
# colors a sequence/list of colors
# stops an optional sequence of fractions 0 - 1
if 4<=len(arg)<=5:
(ax0, ay0), (ax1, ay1) = arg[:2]
ax0 = x0 + ax0*w
ax1 = x0 + ax1*w
ay0 = y0 + ay0*h
ay1 = y0 + ay1*h
extend = arg[2]
C = arg[3]
P = arg[4] if len(arg)==4 else None
canv.linearGradient(ax0, ay0, ax1, ay1, C, positions=P, extend=extend)
else:
raise ValueError(f'Wrong length for {op!a} arguments {arg!a}')
elif arg0=='RADIALGRADIENT':
# the remaining arguments define the centre, radius, extend, colors, stops as
# center = (xc,yc) given as fractions of width / height
# radius = (r,'ref') op in ('width','height','min','max')
# extend = bool or [bool, bool]
# colors a sequence/list of colors
# stops an optional sequence of fractions 0 - 1
if 4<=len(arg)<=5:
xc, yc = arg[0]
xc = x0 + xc*w
yc = y0 + yc*h
r, ref = arg[1]
if ref=='width':
ref = w
elif ref=='height':
ref = h
elif ref=='min':
ref = min (w,h)
elif ref=='max':
ref = max(w,h)
else:
raise ValueError(f'Bad radius, {arg[1]!a}, for {op!a} arguments {arg!r}')
r *= ref
extend = arg[2]
C = arg[3]
P = arg[4] if len(arg)==4 else None
canv.radialGradient(xc, yc, r, C, positions=P, extend=extend)
else:
raise ValueError(f'Wrong length for {op!a} arguments {arg}')
canv.restoreState()
else:
color = colors.toColorOrNone(arg)
if color:
if ec==sc and er==sr and spanRects:
xywh = spanRects.get((sc,sr))
if xywh:
#it's a single cell
x0, y0, w, h = xywh
canv.setFillColor(color)
canv.rect(x0, y0, w, h, stroke=0,fill=1)
def _drawCell(self, cellval, cellstyle, pos, size):
colpos, rowpos = pos
colwidth, rowheight = size
if self._curcellstyle is not cellstyle:
cur = self._curcellstyle
if cur is None or cellstyle.color != cur.color:
self.canv.setFillColor(cellstyle.color)
if cur is None or cellstyle.leading != cur.leading or cellstyle.fontname != cur.fontname or cellstyle.fontsize != cur.fontsize:
self.canv.setFont(cellstyle.fontname, cellstyle.fontsize, cellstyle.leading)
self._curcellstyle = cellstyle
just = cellstyle.alignment
valign = cellstyle.valign
if isinstance(cellval,(tuple,list,Flowable)):
if not isinstance(cellval,(tuple,list)): cellval = (cellval,)
# we assume it's a list of Flowables
W = []
H = []
w, h = self._listCellGeom(cellval,colwidth,cellstyle,W=W, H=H,aH=rowheight)
if valign=='TOP':
y = rowpos + rowheight - cellstyle.topPadding
elif valign=='BOTTOM':
y = rowpos+cellstyle.bottomPadding + h
else:
y = rowpos+(rowheight+cellstyle.bottomPadding-cellstyle.topPadding+h)/2.0
if cellval: y += cellval[0].getSpaceBefore()
for v, w, h in zip(cellval,W,H):
if just=='LEFT': x = colpos+cellstyle.leftPadding
elif just=='RIGHT': x = colpos+colwidth-cellstyle.rightPadding - w
elif just in ('CENTRE', 'CENTER'):
x = colpos+(colwidth+cellstyle.leftPadding-cellstyle.rightPadding-w)/2.0
else:
raise ValueError(f'Invalid justification {just!a} for {type(v)}')
y -= v.getSpaceBefore()
y -= h
v.drawOn(self.canv,x,y)
y -= v.getSpaceAfter()
else:
if just == 'LEFT':
draw = self.canv.drawString
x = colpos + cellstyle.leftPadding
elif just in ('CENTRE', 'CENTER'):
draw = self.canv.drawCentredString
x = colpos+(colwidth+cellstyle.leftPadding-cellstyle.rightPadding)*0.5
elif just == 'RIGHT':
draw = self.canv.drawRightString
x = colpos + colwidth - cellstyle.rightPadding
elif just == 'DECIMAL':
draw = self.canv.drawAlignedString
x = colpos + colwidth - cellstyle.rightPadding
else:
raise ValueError(f'Invalid justification {just!a}')
vals = str(cellval).split("\n")
n = len(vals)
leading = cellstyle.leading
fontsize = cellstyle.fontsize
if valign=='BOTTOM':
y = rowpos + cellstyle.bottomPadding+n*leading-fontsize
elif valign=='TOP':
y = rowpos + rowheight - cellstyle.topPadding - fontsize
elif valign=='MIDDLE':
#tim roberts pointed out missing fontsize correction 2004-10-04
y = rowpos + (cellstyle.bottomPadding + rowheight-cellstyle.topPadding+n*leading)/2.0 - fontsize
else:
raise ValueError(f'Bad valign: {valign!a}')
for v in vals:
draw(x, y, v)
y -= leading
onDraw = getattr(cellval,'onDraw',None)
if onDraw:
onDraw(self.canv,cellval.kind,cellval.label)
if cellstyle.href:
#external hyperlink
self.canv.linkURL(cellstyle.href, (colpos, rowpos, colpos + colwidth, rowpos + rowheight), relative=1)
if cellstyle.destination:
#external hyperlink
self.canv.linkRect("", cellstyle.destination, Rect=(colpos, rowpos, colpos + colwidth, rowpos + rowheight), relative=1)
def _setCornerRadii(self, cornerRadii):
if isListOfNumbersOrNone(cornerRadii):
self._cornerRadii = None if not cornerRadii else list(cornerRadii) + (max(4-len(cornerRadii),0)*[0])
else:
raise ValueError(f'cornerRadii should be None or a list/tuple of numeric radii\nnot {cornerRadii!a}')
_LineOpMap = { 'GRID':'_drawGrid',
'BOX':'_drawBox',
'OUTLINE':'_drawBox',
'INNERGRID':'_drawInnerGrid',
'LINEBELOW':'_drawHLinesB',
'LINEABOVE':'_drawHLines',
'LINEBEFORE':'_drawVLines',
'LINEAFTER':'_drawVLinesA', }
class LongTable(Table):
'''Henning von Bargen's changes will be active'''
_longTableOptimize = 1
LINECOMMANDS = list(_LineOpMap.keys())
class TableRenderCB:
'''table render callback abstract base klass to be called in Table.draw'''
def __call__(self,T,cmd,*args):
if not isinstance(T,Table): raise ValueError(f'TableRenderCB first argument, {repr(T)} is not a Table')
meth = getattr(self,cmd,None)
if not meth: raise ValueError(f'invalid TablerenderCB cmd {cmd}')
meth(T,*args)
def startTable(self,T):
raise NotImplementedError('startTable')
def startBG(self,T):
raise NotImplementedError('startBG')
def endBG(self,T):
raise NotImplementedError('endBG')
def startRow(self,T,rowNo):
raise NotImplementedError('startRow')
def startCell(self,T,rowNo, colNo, cellval, cellstyle, pos, size):
raise NotImplementedError('startCell')
def endCell(self,T):
raise NotImplementedError('endCell')
def endRow(self,T):
raise NotImplementedError('endRow')
def startLines(self,T):
raise NotImplementedError('startLines')
def endLines(self,T):
raise NotImplementedError('endLines')
def endTable(self,T):
raise NotImplementedError('endTable')
def _isLineCommand(cmd):
return cmd[0] in LINECOMMANDS
def _setCellStyle(cellStyles, i, j, op, values):
#new = CellStyle('<%d, %d>' % (i,j), cellStyles[i][j])
#cellStyles[i][j] = new
## modify in place!!!
new = cellStyles[i][j]
if op == 'FONT':
n = len(values)
new.fontname = values[0]
if n>1:
new.fontsize = values[1]
if n>2:
new.leading = values[2]
else:
new.leading = new.fontsize*1.2
elif op in ('FONTNAME', 'FACE'):
new.fontname = values[0]
elif op in ('SIZE', 'FONTSIZE'):
new.fontsize = values[0]
elif op == 'LEADING':
new.leading = values[0]
elif op == 'TEXTCOLOR':
new.color = colors.toColor(values[0], colors.Color(0,0,0))
elif op in ('ALIGN', 'ALIGNMENT'):
new.alignment = values[0]
elif op == 'VALIGN':
new.valign = values[0]
elif op == 'LEFTPADDING':
new.leftPadding = values[0]
elif op == 'RIGHTPADDING':
new.rightPadding = values[0]
elif op == 'TOPPADDING':
new.topPadding = values[0]
elif op == 'BOTTOMPADDING':
new.bottomPadding = values[0]
elif op == 'HREF':
new.href = values[0]
elif op == 'DESTINATION':
new.destination = values[0]
GRID_STYLE = TableStyle(
[('GRID', (0,0), (-1,-1), 0.25, colors.black),
('ALIGN', (1,1), (-1,-1), 'RIGHT')]
)
BOX_STYLE = TableStyle(
[('BOX', (0,0), (-1,-1), 0.50, colors.black),
('ALIGN', (1,1), (-1,-1), 'RIGHT')]
)
LABELED_GRID_STYLE = TableStyle(
[('INNERGRID', (0,0), (-1,-1), 0.25, colors.black),
('BOX', (0,0), (-1,-1), 2, colors.black),
('LINEBELOW', (0,0), (-1,0), 2, colors.black),
('LINEAFTER', (0,0), (0,-1), 2, colors.black),
('ALIGN', (1,1), (-1,-1), 'RIGHT')]
)
COLORED_GRID_STYLE = TableStyle(
[('INNERGRID', (0,0), (-1,-1), 0.25, colors.black),
('BOX', (0,0), (-1,-1), 2, colors.red),
('LINEBELOW', (0,0), (-1,0), 2, colors.black),
('LINEAFTER', (0,0), (0,-1), 2, colors.black),
('ALIGN', (1,1), (-1,-1), 'RIGHT')]
)
LIST_STYLE = TableStyle(
[('LINEABOVE', (0,0), (-1,0), 2, colors.green),
('LINEABOVE', (0,1), (-1,-1), 0.25, colors.black),
('LINEBELOW', (0,-1), (-1,-1), 2, colors.green),
('ALIGN', (1,1), (-1,-1), 'RIGHT')]
)
# experimental iterator which can apply a sequence
# of colors e.g. Blue, None, Blue, None as you move
# down.
if __name__ == '__main__':
from tests.test_platypus_tables import old_tables_test
old_tables_test()