author | rgbecker |
Thu, 27 May 2010 15:51:47 +0000 | |
changeset 3385 | e45ca0b2053c |
parent 3368 | afa025c34493 |
child 3617 | ae5744e97c42 |
permissions | -rw-r--r-- |
2332 | 1 |
#Copyright ReportLab Europe Ltd. 2000-2004 |
817 | 2 |
#see license.txt for license details |
2332 | 3 |
#history http://www.reportlab.co.uk/cgi-bin/viewcvs.cgi/public/reportlab/trunk/reportlab/graphics/renderbase.py |
568 | 4 |
|
5 |
__version__=''' $Id $ ''' |
|
3032 | 6 |
__doc__='''Superclass for renderers to factor out common functionality and default implementations.''' |
568 | 7 |
|
8 |
from reportlab.graphics.shapes import * |
|
2547 | 9 |
from reportlab.lib.validators import DerivedValue |
2360
0fbaee224de1
rl_config add _unset_, graphics.renderxxx refactoring
rgbecker
parents:
2332
diff
changeset
|
10 |
from reportlab import rl_config |
568 | 11 |
|
12 |
def inverse(A): |
|
1677
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
13 |
"For A affine 2D represented as 6vec return 6vec version of A**(-1)" |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
14 |
# I checked this RGB |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
15 |
det = float(A[0]*A[3] - A[2]*A[1]) |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
16 |
R = [A[3]/det, -A[1]/det, -A[2]/det, A[0]/det] |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
17 |
return tuple(R+[-R[0]*A[4]-R[2]*A[5],-R[1]*A[4]-R[3]*A[5]]) |
568 | 18 |
|
19 |
def mmult(A, B): |
|
1677
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
20 |
"A postmultiplied by B" |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
21 |
# I checked this RGB |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
22 |
# [a0 a2 a4] [b0 b2 b4] |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
23 |
# [a1 a3 a5] * [b1 b3 b5] |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
24 |
# [ 1 ] [ 1 ] |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
25 |
# |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
26 |
return (A[0]*B[0] + A[2]*B[1], |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
27 |
A[1]*B[0] + A[3]*B[1], |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
28 |
A[0]*B[2] + A[2]*B[3], |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
29 |
A[1]*B[2] + A[3]*B[3], |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
30 |
A[0]*B[4] + A[2]*B[5] + A[4], |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
31 |
A[1]*B[4] + A[3]*B[5] + A[5]) |
568 | 32 |
|
33 |
||
34 |
def getStateDelta(shape): |
|
1677
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
35 |
"""Used to compute when we need to change the graphics state. |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
36 |
For example, if we have two adjacent red shapes we don't need |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
37 |
to set the pen color to red in between. Returns the effect |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
38 |
the given shape would have on the graphics state""" |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
39 |
delta = {} |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
40 |
for (prop, value) in shape.getProperties().items(): |
3326 | 41 |
if prop in STATE_DEFAULTS: |
1677
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
42 |
delta[prop] = value |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
43 |
return delta |
568 | 44 |
|
45 |
||
46 |
class StateTracker: |
|
1677
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
47 |
"""Keeps a stack of transforms and state |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
48 |
properties. It can contain any properties you |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
49 |
want, but the keys 'transform' and 'ctm' have |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
50 |
special meanings. The getCTM() |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
51 |
method returns the current transformation |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
52 |
matrix at any point, without needing to |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
53 |
invert matrixes when you pop.""" |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
54 |
def __init__(self, defaults=None): |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
55 |
# one stack to keep track of what changes... |
2547 | 56 |
self._deltas = [] |
568 | 57 |
|
1677
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
58 |
# and another to keep track of cumulative effects. Last one in |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
59 |
# list is the current graphics state. We put one in to simplify |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
60 |
# loops below. |
2547 | 61 |
self._combined = [] |
1677
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
62 |
if defaults is None: |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
63 |
defaults = STATE_DEFAULTS.copy() |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
64 |
#ensure that if we have a transform, we have a CTM |
3326 | 65 |
if 'transform' in defaults: |
1677
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
66 |
defaults['ctm'] = defaults['transform'] |
2547 | 67 |
self._combined.append(defaults) |
568 | 68 |
|
1677
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
69 |
def push(self,delta): |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
70 |
"""Take a new state dictionary of changes and push it onto |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
71 |
the stack. After doing this, the combined state is accessible |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
72 |
through getState()""" |
1665 | 73 |
|
2547 | 74 |
newstate = self._combined[-1].copy() |
1677
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
75 |
for (key, value) in delta.items(): |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
76 |
if key == 'transform': #do cumulative matrix |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
77 |
newstate['transform'] = delta['transform'] |
2547 | 78 |
newstate['ctm'] = mmult(self._combined[-1]['ctm'], delta['transform']) |
1677
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
79 |
#print 'statetracker transform = (%0.2f, %0.2f, %0.2f, %0.2f, %0.2f, %0.2f)' % tuple(newstate['transform']) |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
80 |
#print 'statetracker ctm = (%0.2f, %0.2f, %0.2f, %0.2f, %0.2f, %0.2f)' % tuple(newstate['ctm']) |
568 | 81 |
|
1677
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
82 |
else: #just overwrite it |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
83 |
newstate[key] = value |
568 | 84 |
|
2547 | 85 |
self._combined.append(newstate) |
86 |
self._deltas.append(delta) |
|
568 | 87 |
|
1677
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
88 |
def pop(self): |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
89 |
"""steps back one, and returns a state dictionary with the |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
90 |
deltas to reverse out of wherever you are. Depending |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
91 |
on your back end, you may not need the return value, |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
92 |
since you can get the complete state afterwards with getState()""" |
2547 | 93 |
del self._combined[-1] |
94 |
newState = self._combined[-1] |
|
95 |
lastDelta = self._deltas[-1] |
|
96 |
del self._deltas[-1] |
|
1677
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
97 |
#need to diff this against the last one in the state |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
98 |
reverseDelta = {} |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
99 |
#print 'pop()...' |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
100 |
for key, curValue in lastDelta.items(): |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
101 |
#print ' key=%s, value=%s' % (key, curValue) |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
102 |
prevValue = newState[key] |
3326 | 103 |
if prevValue != curValue: |
1677
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
104 |
#print ' state popping "%s"="%s"' % (key, curValue) |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
105 |
if key == 'transform': |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
106 |
reverseDelta[key] = inverse(lastDelta['transform']) |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
107 |
else: #just return to previous state |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
108 |
reverseDelta[key] = prevValue |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
109 |
return reverseDelta |
568 | 110 |
|
1677
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
111 |
def getState(self): |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
112 |
"returns the complete graphics state at this point" |
2547 | 113 |
return self._combined[-1] |
568 | 114 |
|
1677
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
115 |
def getCTM(self): |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
116 |
"returns the current transformation matrix at this point""" |
2547 | 117 |
return self._combined[-1]['ctm'] |
568 | 118 |
|
2401 | 119 |
def __getitem__(self,key): |
120 |
"returns the complete graphics state value of key at this point" |
|
2547 | 121 |
return self._combined[-1][key] |
2401 | 122 |
|
123 |
def __setitem__(self,key,value): |
|
124 |
"sets the complete graphics state value of key to value" |
|
2547 | 125 |
self._combined[-1][key] = value |
568 | 126 |
|
127 |
def testStateTracker(): |
|
1677
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
128 |
print 'Testing state tracker' |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
129 |
defaults = {'fillColor':None, 'strokeColor':None,'fontName':None, 'transform':[1,0,0,1,0,0]} |
3368
afa025c34493
reportlab: new base font mechanism more fully applied
rgbecker
parents:
3326
diff
changeset
|
130 |
from reportlab.graphics.shapes import _baseGFontName |
1677
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
131 |
deltas = [ |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
132 |
{'fillColor':'red'}, |
3368
afa025c34493
reportlab: new base font mechanism more fully applied
rgbecker
parents:
3326
diff
changeset
|
133 |
{'fillColor':'green', 'strokeColor':'blue','fontName':_baseGFontName}, |
1677
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
134 |
{'transform':[0.5,0,0,0.5,0,0]}, |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
135 |
{'transform':[0.5,0,0,0.5,2,3]}, |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
136 |
{'strokeColor':'red'} |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
137 |
] |
1665 | 138 |
|
1677
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
139 |
st = StateTracker(defaults) |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
140 |
print 'initial:', st.getState() |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
141 |
|
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
142 |
for delta in deltas: |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
143 |
print 'pushing:', delta |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
144 |
st.push(delta) |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
145 |
print 'state: ',st.getState(),'\n' |
568 | 146 |
|
1677
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
147 |
for delta in deltas: |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
148 |
print 'popping:',st.pop() |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
149 |
print 'state: ',st.getState(),'\n' |
1665 | 150 |
|
568 | 151 |
|
1665 | 152 |
def _expandUserNode(node,canvas): |
1677
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
153 |
if isinstance(node, UserNode): |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
154 |
try: |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
155 |
if hasattr(node,'_canvas'): |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
156 |
ocanvas = 1 |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
157 |
else: |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
158 |
node._canvas = canvas |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
159 |
ocanvas = None |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
160 |
onode = node |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
161 |
node = node.provideNode() |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
162 |
finally: |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
163 |
if not ocanvas: del onode._canvas |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
164 |
return node |
568 | 165 |
|
2544
a6b9aa99b3c3
graphics: added Drawing.renderScale hack for renderer terminal drawing scales
rgbecker
parents:
2401
diff
changeset
|
166 |
def renderScaledDrawing(d): |
a6b9aa99b3c3
graphics: added Drawing.renderScale hack for renderer terminal drawing scales
rgbecker
parents:
2401
diff
changeset
|
167 |
renderScale = d.renderScale |
a6b9aa99b3c3
graphics: added Drawing.renderScale hack for renderer terminal drawing scales
rgbecker
parents:
2401
diff
changeset
|
168 |
if renderScale!=1.0: |
2827
ec03c767079c
renderbase.py: attempt to fix problems with copy in renderScaledDrawing
rgbecker
parents:
2570
diff
changeset
|
169 |
o = d |
ec03c767079c
renderbase.py: attempt to fix problems with copy in renderScaledDrawing
rgbecker
parents:
2570
diff
changeset
|
170 |
d = d.__class__(o.width*renderScale,o.height*renderScale) |
ec03c767079c
renderbase.py: attempt to fix problems with copy in renderScaledDrawing
rgbecker
parents:
2570
diff
changeset
|
171 |
d.__dict__ = o.__dict__.copy() |
2544
a6b9aa99b3c3
graphics: added Drawing.renderScale hack for renderer terminal drawing scales
rgbecker
parents:
2401
diff
changeset
|
172 |
d.scale(renderScale,renderScale) |
a6b9aa99b3c3
graphics: added Drawing.renderScale hack for renderer terminal drawing scales
rgbecker
parents:
2401
diff
changeset
|
173 |
d.renderScale = 1.0 |
a6b9aa99b3c3
graphics: added Drawing.renderScale hack for renderer terminal drawing scales
rgbecker
parents:
2401
diff
changeset
|
174 |
return d |
a6b9aa99b3c3
graphics: added Drawing.renderScale hack for renderer terminal drawing scales
rgbecker
parents:
2401
diff
changeset
|
175 |
|
568 | 176 |
class Renderer: |
1677
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
177 |
"""Virtual superclass for graphics renderers.""" |
1665 | 178 |
|
1677
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
179 |
def __init__(self): |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
180 |
self._tracker = StateTracker() |
2547 | 181 |
self._nodeStack = [] #track nodes visited |
1665 | 182 |
|
1677
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
183 |
def undefined(self, operation): |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
184 |
raise ValueError, "%s operation not defined at superclass class=%s" %(operation, self.__class__) |
568 | 185 |
|
2360
0fbaee224de1
rl_config add _unset_, graphics.renderxxx refactoring
rgbecker
parents:
2332
diff
changeset
|
186 |
def draw(self, drawing, canvas, x=0, y=0, showBoundary=rl_config._unset_): |
0fbaee224de1
rl_config add _unset_, graphics.renderxxx refactoring
rgbecker
parents:
2332
diff
changeset
|
187 |
"""This is the top level function, which draws the drawing at the given |
0fbaee224de1
rl_config add _unset_, graphics.renderxxx refactoring
rgbecker
parents:
2332
diff
changeset
|
188 |
location. The recursive part is handled by drawNode.""" |
0fbaee224de1
rl_config add _unset_, graphics.renderxxx refactoring
rgbecker
parents:
2332
diff
changeset
|
189 |
#stash references for ease of communication |
0fbaee224de1
rl_config add _unset_, graphics.renderxxx refactoring
rgbecker
parents:
2332
diff
changeset
|
190 |
if showBoundary is rl_config._unset_: showBoundary=rl_config.showBoundary |
0fbaee224de1
rl_config add _unset_, graphics.renderxxx refactoring
rgbecker
parents:
2332
diff
changeset
|
191 |
self._canvas = canvas |
0fbaee224de1
rl_config add _unset_, graphics.renderxxx refactoring
rgbecker
parents:
2332
diff
changeset
|
192 |
canvas.__dict__['_drawing'] = self._drawing = drawing |
0fbaee224de1
rl_config add _unset_, graphics.renderxxx refactoring
rgbecker
parents:
2332
diff
changeset
|
193 |
drawing._parent = None |
0fbaee224de1
rl_config add _unset_, graphics.renderxxx refactoring
rgbecker
parents:
2332
diff
changeset
|
194 |
try: |
0fbaee224de1
rl_config add _unset_, graphics.renderxxx refactoring
rgbecker
parents:
2332
diff
changeset
|
195 |
#bounding box |
0fbaee224de1
rl_config add _unset_, graphics.renderxxx refactoring
rgbecker
parents:
2332
diff
changeset
|
196 |
if showBoundary: canvas.rect(x, y, drawing.width, drawing.height) |
0fbaee224de1
rl_config add _unset_, graphics.renderxxx refactoring
rgbecker
parents:
2332
diff
changeset
|
197 |
canvas.saveState() |
2547 | 198 |
self.initState(x,y) #this is the push() |
2360
0fbaee224de1
rl_config add _unset_, graphics.renderxxx refactoring
rgbecker
parents:
2332
diff
changeset
|
199 |
self.drawNode(drawing) |
0fbaee224de1
rl_config add _unset_, graphics.renderxxx refactoring
rgbecker
parents:
2332
diff
changeset
|
200 |
self.pop() |
0fbaee224de1
rl_config add _unset_, graphics.renderxxx refactoring
rgbecker
parents:
2332
diff
changeset
|
201 |
canvas.restoreState() |
0fbaee224de1
rl_config add _unset_, graphics.renderxxx refactoring
rgbecker
parents:
2332
diff
changeset
|
202 |
finally: |
0fbaee224de1
rl_config add _unset_, graphics.renderxxx refactoring
rgbecker
parents:
2332
diff
changeset
|
203 |
#remove any circular references |
0fbaee224de1
rl_config add _unset_, graphics.renderxxx refactoring
rgbecker
parents:
2332
diff
changeset
|
204 |
del self._canvas, self._drawing, canvas._drawing, drawing._parent |
0fbaee224de1
rl_config add _unset_, graphics.renderxxx refactoring
rgbecker
parents:
2332
diff
changeset
|
205 |
|
0fbaee224de1
rl_config add _unset_, graphics.renderxxx refactoring
rgbecker
parents:
2332
diff
changeset
|
206 |
def initState(self,x,y): |
0fbaee224de1
rl_config add _unset_, graphics.renderxxx refactoring
rgbecker
parents:
2332
diff
changeset
|
207 |
deltas = STATE_DEFAULTS.copy() |
0fbaee224de1
rl_config add _unset_, graphics.renderxxx refactoring
rgbecker
parents:
2332
diff
changeset
|
208 |
deltas['transform'] = [1,0,0,1,x,y] |
0fbaee224de1
rl_config add _unset_, graphics.renderxxx refactoring
rgbecker
parents:
2332
diff
changeset
|
209 |
self._tracker.push(deltas) |
0fbaee224de1
rl_config add _unset_, graphics.renderxxx refactoring
rgbecker
parents:
2332
diff
changeset
|
210 |
self.applyStateChanges(deltas, {}) |
0fbaee224de1
rl_config add _unset_, graphics.renderxxx refactoring
rgbecker
parents:
2332
diff
changeset
|
211 |
|
0fbaee224de1
rl_config add _unset_, graphics.renderxxx refactoring
rgbecker
parents:
2332
diff
changeset
|
212 |
def pop(self): |
0fbaee224de1
rl_config add _unset_, graphics.renderxxx refactoring
rgbecker
parents:
2332
diff
changeset
|
213 |
self._tracker.pop() |
568 | 214 |
|
1677
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
215 |
def drawNode(self, node): |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
216 |
"""This is the recursive method called for each node |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
217 |
in the tree""" |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
218 |
# Undefined here, but with closer analysis probably can be handled in superclass |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
219 |
self.undefined("drawNode") |
1665 | 220 |
|
2547 | 221 |
def getStateValue(self, key): |
222 |
"""Return current state parameter for given key""" |
|
223 |
currentState = self._tracker._combined[-1] |
|
224 |
return currentState[key] |
|
225 |
||
226 |
def fillDerivedValues(self, node): |
|
227 |
"""Examine a node for any values which are Derived, |
|
228 |
and replace them with their calculated values. |
|
229 |
Generally things may look at the drawing or their |
|
230 |
parent. |
|
231 |
||
232 |
""" |
|
233 |
for (key, value) in node.__dict__.items(): |
|
234 |
if isinstance(value, DerivedValue): |
|
235 |
#just replace with default for key? |
|
236 |
#print ' fillDerivedValues(%s)' % key |
|
237 |
newValue = value.getValue(self, key) |
|
238 |
#print ' got value of %s' % newValue |
|
239 |
node.__dict__[key] = newValue |
|
240 |
||
1677
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
241 |
def drawNodeDispatcher(self, node): |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
242 |
"""dispatch on the node's (super) class: shared code""" |
2570
7b93b3b42e51
reportlab/graphics cosmetics and adding drawImage to renderPS
rgbecker
parents:
2547
diff
changeset
|
243 |
|
1677
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
244 |
canvas = getattr(self,'_canvas',None) |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
245 |
# replace UserNode with its contents |
2570
7b93b3b42e51
reportlab/graphics cosmetics and adding drawImage to renderPS
rgbecker
parents:
2547
diff
changeset
|
246 |
|
1677
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
247 |
try: |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
248 |
node = _expandUserNode(node,canvas) |
3233 | 249 |
if not node: return |
1677
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
250 |
if hasattr(node,'_canvas'): |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
251 |
ocanvas = 1 |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
252 |
else: |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
253 |
node._canvas = canvas |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
254 |
ocanvas = None |
568 | 255 |
|
2547 | 256 |
self.fillDerivedValues(node) |
3385 | 257 |
dtcb = getattr(node,'_drawTimeCallback',None) |
258 |
if dtcb: |
|
259 |
dtcb(node,canvas=canvas,renderer=self) |
|
1677
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
260 |
#draw the object, or recurse |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
261 |
if isinstance(node, Line): |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
262 |
self.drawLine(node) |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
263 |
elif isinstance(node, Image): |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
264 |
self.drawImage(node) |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
265 |
elif isinstance(node, Rect): |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
266 |
self.drawRect(node) |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
267 |
elif isinstance(node, Circle): |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
268 |
self.drawCircle(node) |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
269 |
elif isinstance(node, Ellipse): |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
270 |
self.drawEllipse(node) |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
271 |
elif isinstance(node, PolyLine): |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
272 |
self.drawPolyLine(node) |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
273 |
elif isinstance(node, Polygon): |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
274 |
self.drawPolygon(node) |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
275 |
elif isinstance(node, Path): |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
276 |
self.drawPath(node) |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
277 |
elif isinstance(node, String): |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
278 |
self.drawString(node) |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
279 |
elif isinstance(node, Group): |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
280 |
self.drawGroup(node) |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
281 |
elif isinstance(node, Wedge): |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
282 |
self.drawWedge(node) |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
283 |
else: |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
284 |
print 'DrawingError','Unexpected element %s in drawing!' % str(node) |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
285 |
finally: |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
286 |
if not ocanvas: del node._canvas |
1665 | 287 |
|
1677
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
288 |
_restores = {'stroke':'_stroke','stroke_width': '_lineWidth','stroke_linecap':'_lineCap', |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
289 |
'stroke_linejoin':'_lineJoin','fill':'_fill','font_family':'_font', |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
290 |
'font_size':'_fontSize'} |
585
e0144950b3e2
Fixes to CTM to support bitmap renderer; extra string rotation
andy_robinson
parents:
580
diff
changeset
|
291 |
|
1677
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
292 |
def drawGroup(self, group): |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
293 |
# just do the contents. Some renderers might need to override this |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
294 |
# if they need a flipped transform |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
295 |
canvas = getattr(self,'_canvas',None) |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
296 |
for node in group.getContents(): |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
297 |
node = _expandUserNode(node,canvas) |
3233 | 298 |
if not node: continue |
2547 | 299 |
|
300 |
#here is where we do derived values - this seems to get everything. Touch wood. |
|
301 |
self.fillDerivedValues(node) |
|
1677
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
302 |
try: |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
303 |
if hasattr(node,'_canvas'): |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
304 |
ocanvas = 1 |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
305 |
else: |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
306 |
node._canvas = canvas |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
307 |
ocanvas = None |
2360
0fbaee224de1
rl_config add _unset_, graphics.renderxxx refactoring
rgbecker
parents:
2332
diff
changeset
|
308 |
node._parent = group |
1677
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
309 |
self.drawNode(node) |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
310 |
finally: |
2360
0fbaee224de1
rl_config add _unset_, graphics.renderxxx refactoring
rgbecker
parents:
2332
diff
changeset
|
311 |
del node._parent |
1677
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
312 |
if not ocanvas: del node._canvas |
585
e0144950b3e2
Fixes to CTM to support bitmap renderer; extra string rotation
andy_robinson
parents:
580
diff
changeset
|
313 |
|
1677
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
314 |
def drawWedge(self, wedge): |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
315 |
# by default ask the wedge to make a polygon of itself and draw that! |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
316 |
#print "drawWedge" |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
317 |
polygon = wedge.asPolygon() |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
318 |
self.drawPolygon(polygon) |
1665 | 319 |
|
1677
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
320 |
def drawPath(self, path): |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
321 |
polygons = path.asPolygons() |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
322 |
for polygon in polygons: |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
323 |
self.drawPolygon(polygon) |
568 | 324 |
|
1677
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
325 |
def drawRect(self, rect): |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
326 |
# could be implemented in terms of polygon |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
327 |
self.undefined("drawRect") |
568 | 328 |
|
1677
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
329 |
def drawLine(self, line): |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
330 |
self.undefined("drawLine") |
568 | 331 |
|
1677
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
332 |
def drawCircle(self, circle): |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
333 |
self.undefined("drawCircle") |
1665 | 334 |
|
1677
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
335 |
def drawPolyLine(self, p): |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
336 |
self.undefined("drawPolyLine") |
568 | 337 |
|
1677
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
338 |
def drawEllipse(self, ellipse): |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
339 |
self.undefined("drawEllipse") |
568 | 340 |
|
1677
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
341 |
def drawPolygon(self, p): |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
342 |
self.undefined("drawPolygon") |
568 | 343 |
|
1677
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
344 |
def drawString(self, stringObj): |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
345 |
self.undefined("drawString") |
568 | 346 |
|
1677
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
347 |
def applyStateChanges(self, delta, newState): |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
348 |
"""This takes a set of states, and outputs the operators |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
349 |
needed to set those properties""" |
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
350 |
self.undefined("applyStateChanges") |
568 | 351 |
|
352 |
if __name__=='__main__': |
|
1677
1450177dd19e
Exterminated all tab characters and added a test to make sure
andy_robinson
parents:
1665
diff
changeset
|
353 |
print "this file has no script interpretation" |
2360
0fbaee224de1
rl_config add _unset_, graphics.renderxxx refactoring
rgbecker
parents:
2332
diff
changeset
|
354 |
print __doc__ |