blob: e915ce939ad0e7337c71524b190595d620046857 [file] [log] [blame]
Guido van Rossumb241b671998-12-04 16:42:46 +00001# LogoMation-like turtle graphics
2
3from math import * # Also for export
4import Tkinter
Guido van Rossumfd2ede22002-09-23 16:55:05 +00005
Martin v. Löwis4b6ea792000-10-01 17:52:01 +00006class Error(Exception):
7 pass
Guido van Rossumb241b671998-12-04 16:42:46 +00008
9class RawPen:
10
11 def __init__(self, canvas):
12 self._canvas = canvas
13 self._items = []
14 self._tracing = 1
Guido van Rossum3c7a25a2001-08-09 16:42:07 +000015 self._arrow = 0
Guido van Rossumb241b671998-12-04 16:42:46 +000016 self.degrees()
17 self.reset()
18
19 def degrees(self, fullcircle=360.0):
20 self._fullcircle = fullcircle
21 self._invradian = pi / (fullcircle * 0.5)
22
23 def radians(self):
24 self.degrees(2.0*pi)
25
26 def reset(self):
27 canvas = self._canvas
Martin v. Löwis73b9b662002-09-22 13:00:26 +000028 self._canvas.update()
Guido van Rossumb241b671998-12-04 16:42:46 +000029 width = canvas.winfo_width()
30 height = canvas.winfo_height()
31 if width <= 1:
32 width = canvas['width']
33 if height <= 1:
34 height = canvas['height']
35 self._origin = float(width)/2.0, float(height)/2.0
36 self._position = self._origin
37 self._angle = 0.0
38 self._drawing = 1
39 self._width = 1
40 self._color = "black"
41 self._filling = 0
42 self._path = []
43 self._tofill = []
44 self.clear()
45 canvas._root().tkraise()
46
47 def clear(self):
48 self.fill(0)
49 canvas = self._canvas
50 items = self._items
51 self._items = []
52 for item in items:
53 canvas.delete(item)
Guido van Rossum3c7a25a2001-08-09 16:42:07 +000054 self._delete_turtle()
55 self._draw_turtle()
56
Guido van Rossumb241b671998-12-04 16:42:46 +000057 def tracer(self, flag):
58 self._tracing = flag
Guido van Rossum3c7a25a2001-08-09 16:42:07 +000059 if not self._tracing:
60 self._delete_turtle()
61 self._draw_turtle()
Guido van Rossumb241b671998-12-04 16:42:46 +000062
63 def forward(self, distance):
64 x0, y0 = start = self._position
65 x1 = x0 + distance * cos(self._angle*self._invradian)
66 y1 = y0 - distance * sin(self._angle*self._invradian)
67 self._goto(x1, y1)
68
69 def backward(self, distance):
70 self.forward(-distance)
71
72 def left(self, angle):
73 self._angle = (self._angle + angle) % self._fullcircle
Guido van Rossum3c7a25a2001-08-09 16:42:07 +000074 self._draw_turtle()
Guido van Rossumb241b671998-12-04 16:42:46 +000075
76 def right(self, angle):
77 self.left(-angle)
78
79 def up(self):
80 self._drawing = 0
81
82 def down(self):
83 self._drawing = 1
84
85 def width(self, width):
86 self._width = float(width)
87
88 def color(self, *args):
89 if not args:
90 raise Error, "no color arguments"
91 if len(args) == 1:
92 color = args[0]
93 if type(color) == type(""):
94 # Test the color first
95 try:
96 id = self._canvas.create_line(0, 0, 0, 0, fill=color)
Martin v. Löwis4b6ea792000-10-01 17:52:01 +000097 except Tkinter.TclError:
Guido van Rossumb241b671998-12-04 16:42:46 +000098 raise Error, "bad color string: %s" % `color`
Guido van Rossum3c7a25a2001-08-09 16:42:07 +000099 self._set_color(color)
Guido van Rossumb241b671998-12-04 16:42:46 +0000100 return
101 try:
102 r, g, b = color
103 except:
104 raise Error, "bad color sequence: %s" % `color`
105 else:
106 try:
107 r, g, b = args
108 except:
109 raise Error, "bad color arguments: %s" % `args`
110 assert 0 <= r <= 1
111 assert 0 <= g <= 1
112 assert 0 <= b <= 1
113 x = 255.0
114 y = 0.5
Guido van Rossum3c7a25a2001-08-09 16:42:07 +0000115 self._set_color("#%02x%02x%02x" % (int(r*x+y), int(g*x+y), int(b*x+y)))
116
117 def _set_color(self,color):
118 self._color = color
119 self._draw_turtle()
120
Guido van Rossumb241b671998-12-04 16:42:46 +0000121 def write(self, arg, move=0):
122 x, y = start = self._position
123 x = x-1 # correction -- calibrated for Windows
Fred Draked038ca82000-10-23 18:31:14 +0000124 item = self._canvas.create_text(x, y,
Guido van Rossumb241b671998-12-04 16:42:46 +0000125 text=str(arg), anchor="sw",
126 fill=self._color)
127 self._items.append(item)
128 if move:
129 x0, y0, x1, y1 = self._canvas.bbox(item)
130 self._goto(x1, y1)
Guido van Rossum3c7a25a2001-08-09 16:42:07 +0000131 self._draw_turtle()
Guido van Rossumb241b671998-12-04 16:42:46 +0000132
133 def fill(self, flag):
134 if self._filling:
135 path = tuple(self._path)
136 smooth = self._filling < 0
137 if len(path) > 2:
138 item = self._canvas._create('polygon', path,
139 {'fill': self._color,
140 'smooth': smooth})
141 self._items.append(item)
142 self._canvas.lower(item)
143 if self._tofill:
144 for item in self._tofill:
145 self._canvas.itemconfigure(item, fill=self._color)
146 self._items.append(item)
147 self._path = []
148 self._tofill = []
149 self._filling = flag
150 if flag:
151 self._path.append(self._position)
152
153 def circle(self, radius, extent=None):
154 if extent is None:
155 extent = self._fullcircle
156 x0, y0 = self._position
157 xc = x0 - radius * sin(self._angle * self._invradian)
158 yc = y0 - radius * cos(self._angle * self._invradian)
159 if radius >= 0.0:
160 start = self._angle - 90.0
161 else:
162 start = self._angle + 90.0
163 extent = -extent
164 if self._filling:
165 if abs(extent) >= self._fullcircle:
166 item = self._canvas.create_oval(xc-radius, yc-radius,
167 xc+radius, yc+radius,
168 width=self._width,
169 outline="")
170 self._tofill.append(item)
171 item = self._canvas.create_arc(xc-radius, yc-radius,
172 xc+radius, yc+radius,
173 style="chord",
174 start=start,
175 extent=extent,
176 width=self._width,
177 outline="")
178 self._tofill.append(item)
179 if self._drawing:
180 if abs(extent) >= self._fullcircle:
181 item = self._canvas.create_oval(xc-radius, yc-radius,
182 xc+radius, yc+radius,
183 width=self._width,
184 outline=self._color)
185 self._items.append(item)
186 item = self._canvas.create_arc(xc-radius, yc-radius,
187 xc+radius, yc+radius,
188 style="arc",
189 start=start,
190 extent=extent,
191 width=self._width,
192 outline=self._color)
193 self._items.append(item)
194 angle = start + extent
195 x1 = xc + abs(radius) * cos(angle * self._invradian)
196 y1 = yc - abs(radius) * sin(angle * self._invradian)
197 self._angle = (self._angle + extent) % self._fullcircle
198 self._position = x1, y1
199 if self._filling:
200 self._path.append(self._position)
Martin v. Löwis73b9b662002-09-22 13:00:26 +0000201 self._draw_turtle()
Guido van Rossumbffa52f2002-09-29 00:25:51 +0000202
Guido van Rossumfd2ede22002-09-23 16:55:05 +0000203 def heading(self):
204 return self._angle
205
206 def setheading(self, angle):
207 self._angle = angle
208 self._draw_turtle()
209
210 def window_width(self):
211 width = self._canvas.winfo_width()
Guido van Rossumbffa52f2002-09-29 00:25:51 +0000212 if width <= 1: # the window isn't managed by a geometry manager
Guido van Rossumfd2ede22002-09-23 16:55:05 +0000213 width = self._canvas['width']
214 return width
215
216 def window_height(self):
217 height = self._canvas.winfo_height()
Guido van Rossumbffa52f2002-09-29 00:25:51 +0000218 if height <= 1: # the window isn't managed by a geometry manager
Guido van Rossumfd2ede22002-09-23 16:55:05 +0000219 height = self._canvas['height']
220 return height
221
222 def position(self):
223 x0, y0 = self._origin
224 x1, y1 = self._position
225 return [x1-x0, -y1+y0]
226
227 def setx(self, xpos):
228 x0, y0 = self._origin
229 x1, y1 = self._position
230 self._goto(x0+xpos, y1)
231
232 def sety(self, ypos):
233 x0, y0 = self._origin
234 x1, y1 = self._position
235 self._goto(x1, y0-ypos)
Guido van Rossumb241b671998-12-04 16:42:46 +0000236
237 def goto(self, *args):
238 if len(args) == 1:
239 try:
240 x, y = args[0]
241 except:
242 raise Error, "bad point argument: %s" % `args[0]`
243 else:
244 try:
245 x, y = args
246 except:
247 raise Error, "bad coordinates: %s" % `args[0]`
248 x0, y0 = self._origin
249 self._goto(x0+x, y0-y)
250
251 def _goto(self, x1, y1):
252 x0, y0 = start = self._position
253 self._position = map(float, (x1, y1))
254 if self._filling:
255 self._path.append(self._position)
256 if self._drawing:
Guido van Rossumbffa52f2002-09-29 00:25:51 +0000257 if self._tracing:
Guido van Rossumb241b671998-12-04 16:42:46 +0000258 dx = float(x1 - x0)
259 dy = float(y1 - y0)
260 distance = hypot(dx, dy)
261 nhops = int(distance)
262 item = self._canvas.create_line(x0, y0, x0, y0,
263 width=self._width,
Guido van Rossumb241b671998-12-04 16:42:46 +0000264 capstyle="round",
265 fill=self._color)
266 try:
267 for i in range(1, 1+nhops):
268 x, y = x0 + dx*i/nhops, y0 + dy*i/nhops
269 self._canvas.coords(item, x0, y0, x, y)
Guido van Rossum3c7a25a2001-08-09 16:42:07 +0000270 self._draw_turtle((x,y))
Guido van Rossumb241b671998-12-04 16:42:46 +0000271 self._canvas.update()
272 self._canvas.after(10)
Guido van Rossuma659efe2001-01-01 19:11:07 +0000273 # in case nhops==0
274 self._canvas.coords(item, x0, y0, x1, y1)
Guido van Rossumb241b671998-12-04 16:42:46 +0000275 self._canvas.itemconfigure(item, arrow="none")
Martin v. Löwis4b6ea792000-10-01 17:52:01 +0000276 except Tkinter.TclError:
Guido van Rossumb241b671998-12-04 16:42:46 +0000277 # Probably the window was closed!
278 return
279 else:
280 item = self._canvas.create_line(x0, y0, x1, y1,
281 width=self._width,
282 capstyle="round",
283 fill=self._color)
284 self._items.append(item)
Guido van Rossum3c7a25a2001-08-09 16:42:07 +0000285 self._draw_turtle()
286
287 def _draw_turtle(self,position=[]):
288 if not self._tracing:
289 return
290 if position == []:
291 position = self._position
292 x,y = position
293 distance = 8
294 dx = distance * cos(self._angle*self._invradian)
295 dy = distance * sin(self._angle*self._invradian)
296 self._delete_turtle()
Martin v. Löwis4157ffb2002-03-28 15:45:57 +0000297 self._arrow = self._canvas.create_line(x-dx,y+dy,x,y,
Guido van Rossum3c7a25a2001-08-09 16:42:07 +0000298 width=self._width,
299 arrow="last",
300 capstyle="round",
301 fill=self._color)
302 self._canvas.update()
303
304 def _delete_turtle(self):
305 if self._arrow != 0:
306 self._canvas.delete(self._arrow)
307 self._arrow = 0
308
Guido van Rossumb241b671998-12-04 16:42:46 +0000309
310
311_root = None
312_canvas = None
313_pen = None
314
315class Pen(RawPen):
316
317 def __init__(self):
318 global _root, _canvas
319 if _root is None:
Martin v. Löwis4b6ea792000-10-01 17:52:01 +0000320 _root = Tkinter.Tk()
Guido van Rossumb241b671998-12-04 16:42:46 +0000321 _root.wm_protocol("WM_DELETE_WINDOW", self._destroy)
322 if _canvas is None:
323 # XXX Should have scroll bars
Martin v. Löwis4b6ea792000-10-01 17:52:01 +0000324 _canvas = Tkinter.Canvas(_root, background="white")
Guido van Rossumb241b671998-12-04 16:42:46 +0000325 _canvas.pack(expand=1, fill="both")
326 RawPen.__init__(self, _canvas)
327
328 def _destroy(self):
329 global _root, _canvas, _pen
330 root = self._canvas._root()
331 if root is _root:
332 _pen = None
333 _root = None
334 _canvas = None
335 root.destroy()
Fred Draked038ca82000-10-23 18:31:14 +0000336
Guido van Rossumb241b671998-12-04 16:42:46 +0000337
338def _getpen():
339 global _pen
340 pen = _pen
341 if not pen:
342 _pen = pen = Pen()
343 return pen
344
345def degrees(): _getpen().degrees()
346def radians(): _getpen().radians()
347def reset(): _getpen().reset()
348def clear(): _getpen().clear()
349def tracer(flag): _getpen().tracer(flag)
350def forward(distance): _getpen().forward(distance)
351def backward(distance): _getpen().backward(distance)
352def left(angle): _getpen().left(angle)
353def right(angle): _getpen().right(angle)
354def up(): _getpen().up()
355def down(): _getpen().down()
356def width(width): _getpen().width(width)
Raymond Hettingerff41c482003-04-06 09:01:11 +0000357def color(*args): _getpen().color(*args)
Guido van Rossumb241b671998-12-04 16:42:46 +0000358def write(arg, move=0): _getpen().write(arg, move)
359def fill(flag): _getpen().fill(flag)
360def circle(radius, extent=None): _getpen().circle(radius, extent)
Raymond Hettingerff41c482003-04-06 09:01:11 +0000361def goto(*args): _getpen().goto(*args)
Guido van Rossumfd2ede22002-09-23 16:55:05 +0000362def heading(): return _getpen().heading()
363def setheading(angle): _getpen().setheading(angle)
364def position(): return _getpen().position()
365def window_width(): return _getpen().window_width()
366def window_height(): return _getpen().window_height()
367def setx(xpos): _getpen().setx(xpos)
368def sety(ypos): _getpen().sety(ypos)
Guido van Rossumb241b671998-12-04 16:42:46 +0000369
370def demo():
371 reset()
372 tracer(1)
373 up()
374 backward(100)
375 down()
376 # draw 3 squares; the last filled
377 width(3)
378 for i in range(3):
379 if i == 2:
380 fill(1)
381 for j in range(4):
382 forward(20)
383 left(90)
384 if i == 2:
385 color("maroon")
386 fill(0)
387 up()
388 forward(30)
389 down()
390 width(1)
391 color("black")
392 # move out of the way
393 tracer(0)
394 up()
395 right(90)
396 forward(100)
397 right(90)
398 forward(100)
399 right(180)
400 down()
401 # some text
402 write("startstart", 1)
403 write("start", 1)
404 color("red")
405 # staircase
406 for i in range(5):
407 forward(20)
408 left(90)
409 forward(20)
410 right(90)
411 # filled staircase
412 fill(1)
413 for i in range(5):
414 forward(20)
415 left(90)
416 forward(20)
417 right(90)
418 fill(0)
419 # more text
420 write("end")
421 if __name__ == '__main__':
422 _root.mainloop()
423
424if __name__ == '__main__':
425 demo()