Python for Designers

by Roberto Arista

Fork me on GitHub

Coordinates and Primitives

abstract-4.svg
graphPaper.svg

A Draw­Bot pro­gram can con­trol the draw­ing of mul­ti­ple PDF can­vases. Think of the can­vas as a sheet of graph pa­per. Draw­Bot is a friend who is will­ing to draw on the graph pa­per ac­cord­ing to your in­struc­tions. The most un­am­bigu­ous way to com­mu­ni­cate what you need is to pro­vide nu­mer­i­cal data to de­scribe shapes: these are co­or­di­nates.

As you launch Draw­Bot, it shows you an empty gray can­vas (left), a text box where you can type code (top right) and a con­sole (bot­tom right). Draw­Bot has a de­fault can­vas size of 1000 by 1000 points. We can de­fine a dif­fer­ent can­vas size us­ing the newPage() func­tion. If you want to work with a spe­cific can­vas size, you should de­fine it be­fore in­vok­ing any other draw­ing com­mand; oth­er­wise, Draw­Bot will pro­vide a stan­dard one. Just like in the phys­i­cal world, where we pick up a piece of pa­per and then we start to draw.

Dif­fer­ently, from an in­ter­ac­tive in­ter­preter, Draw­Bot is script based. It means that you can store your pro­grams as text files with a .py ex­ten­sion. If you want to ex­e­cute your pro­gram, you have to in­voke the in­ter­preter to “run” us­ing cmd+R ex­plic­itly.

Given the two-di­men­sional na­ture of the PDF can­vas you have to pro­vide a pair of co­or­di­nates: x and y. The ori­gin of the co­or­di­nate sys­tem is lo­cated in the lower left cor­ner of the can­vas.

cartesian.svg

The or­der of the co­or­di­nates is con­ven­tion­ally x fol­lowed by y. In other words, width (x) comes first and then height (y).

Take into ac­count that un­like a pixel grid, a vec­tor can­vas is con­tin­u­ous. This means that it can ac­cept float­ing points co­or­di­nates with­out round­ings such as 1.3.

rounding.svg

The newPage() func­tion ac­cepts two kind of pa­ra­me­ters:

  • a pair of coordinates
  • a string with a page format standard from this list.

Since we will dive into strings in the next chap­ters, for now let’s stick to nu­mer­i­cal val­ues.

drawbot_canvas_1.svg

.py

newPage(100, 100)
drawbot_canvas_2.svg

.py

newPage(100, 200)
drawbot_canvas_3.svg

.py

newPage(300, 20)

I as­sume width and height are straight­for­ward no­tions, but, what do these num­bers mean? Ap­ples, me­ters, yards? Well, each di­men­sion pro­vided to any draw­ing func­tion in Draw­Bot is ex­pressed in ty­po­graph­i­cal Post­Script points. As you maybe know from your graphic de­sign his­tory classes, ty­po­graph­i­cal points have a quite trou­ble­some re­la­tion with other unit sys­tems. Since the desk­top pub­lish­ing rev­o­lu­tion, the ty­po­graphic points have been uni­vo­cally made pro­por­tional to an­other unit mea­sure, the inch. One ty­po­graph­i­cal point is equiv­a­lent to the 72nd part of an inch. As­sume we are in the realm of pixel im­ages and the den­sity of the im­age is 72 dpi; then one ty­po­graphic point is also equiv­a­lent to one pixel. We will talk more about im­age res­o­lu­tions in the next chap­ters.

thumbmeter.svg

You should see Draw­Bot as a skilled and quick draughts­man ex­e­cut­ing the in­struc­tions you gave in the code ed­i­tor. As any draw­ing ac­tiv­ity, you should think of col­ors and tools be­fore ac­tu­ally us­ing them. You first de­cide which color, then you start to spread ink on pa­per, right?

glasses.svg

Draw­Bot sets some stan­dard val­ues as soon as you start the ap­pli­ca­tion: white back­ground for the can­vas, RGB black for the fill color, trans­par­ent stroke of width 1pt. We will dive into col­ors more in de­tail in the up­com­ing chap­ters. For the fol­low­ing demos we will use shades of gray, which are ex­pressed in val­ues be­tween 0 and 1:

  • 0 means black
  • 1 means white
  • 0.5 half gray

and so on. Think of these fig­ures as per­cent­ages of light pres­ence (0 as 0%, .5 as 50%, 1 as 100%)

The fea­tures of our draw­ing tool are de­fined by 3 func­tions:

  • fill(aColor)
  • stroke(aColor)
  • strokeWidth(thickness)

They need to be called be­fore the draw­ing func­tions: first we choose the tool, then we draw.

Draw­Bot pro­vides four func­tions for draw­ing prim­i­tive shapes.

rect().svg

.py

rect(x, y, width, height)

This func­tion draws a rec­tan­gle on the can­vas us­ing x and y as lower left cor­ner.

rect.png

.py

newPage(100, 100)

rect(10, 10, 20, 20)
rect(120, 10, 20, 20)  # outside the canvas
rect(40, 10, 10, 30)

ex­er­cise 5.1

Cre­ate a 100x100pt can­vas. Draw four squares, side 20 units, each one fac­ing a cor­ner of the can­vas

>>> Solution (.py)

esRects.png
oval().svg

.py

oval(x, y, width, height)

This func­tion draws an oval on the can­vas us­ing x and y as the lower left point of the rec­tan­gle where you could in­scribe the oval. The width ar­gu­ment and the height ar­gu­ment cor­re­spond to the hor­i­zon­tal and ver­ti­cal di­am­e­ter.

ovals.png

.py

newPage(100, 100)
oval(10, 10, 25, 18)
oval(22, 50, 35, 35) # a circle!
oval(68, 18, 22, 55)

ex­er­cise 5.2

Cre­ate a 100x100pt can­vas. Draw four ovals, di­am­e­ter 20 units, each one hav­ing its cen­ter point to a can­vas cor­ner

>>> Solution (.py)

esOvals.png
line().svg

.py

line((x1, y1), (x2, y2))

This func­tion draws a line be­tween two points. Re­mem­ber to en­close each pair of co­or­di­nates be­tween paren­the­sis.

lines-01.png

.py

newPage(100, 100)
stroke(0)
strokeWidth(2)
# three parallel horizontal lines
# their points share the x values
line((10, 20), (90, 20))
line((10, 50), (90, 50))
line((10, 80), (90, 80))

ex­er­cise 5.3

How could you im­prove the qual­ity of the code adding two iden­ti­fiers?

>>> Solution (.py)

lines2-01.png

.py

newPage(100, 100)
stroke(0)
strokeWidth(2)
# three parallel vertical lines
# their points share the y values
line((20, 10), (20, 90))
line((50, 10), (50, 90))
line((80, 10), (80, 90))
lines3-01.png

.py

newPage(100, 100)
stroke(0)
strokeWidth(2)
# three diagonal lines
# they do not share any value
line((74, 50), (15, 30))
line((42, 63), ( 9, 89))
line((47, 29), (97,  6))

ex­er­cise 5.4

Cre­ate a 100x100pt can­vas. Draw two lines con­nect­ing two non-con­tigu­ous can­vas cor­ners

>>> Solution (.py)

esX.png
lines4-01.png

.py

newPage(100, 100)
stroke(0)
strokeWidth(2)
# three diagonal lines
# they share some points
# they are connected
line((74, 20), (15, 30))
line((15, 30), ( 9, 89))
line(( 9, 89), (97, 46))

ex­er­cise 5.5

Cre­ate a 100x100pt can­vas. Draw a zig zag con­nected poly­line start­ing from top left cor­ner and end­ing in bot­tom right cor­ner of the can­vas

>>> Solution (.py)

esZigZig.png
polygon().svg

.py

polygon((x1, y1), (x2, y2), (x3, y3),... close=True)
polygon.png

.py

newPage(100, 100)
polygon((10, 18), (25, 90), (84, 34))

ex­er­cise 5.6

Draw two poly­gons of three sides each one, each poly­gon should have two ver­tices match­ing two con­tigu­ous can­vas cor­ners and one ver­tex match­ing the can­vas cen­tre

>>> Solution (.py)

esBlackWhite.png

The or­der of ex­e­cu­tion of the code, i.e. the arrange­ment of your state­ments, re­flects the or­der in which el­e­ments will be drawn onto the can­vas. For ex­am­ple:

rect_oval.png

.py

newPage(100, 100)
fill(1)
stroke(0)
strokeWidth(2)
rect(14, 40, 50, 50)
oval(40, 20, 50, 50)

gives a dif­fer­ent re­sult from:

oval_rect.png

.py

newPage(100, 100)
fill(1)
stroke(0)
strokeWidth(2)
oval(40, 20, 50, 50)
rect(14, 40, 50, 50)

ex­er­cise 5.7

Cre­ate a 100x100pt can­vas. Com­bine the cir­cles and the squares ex­er­cises, but draw the four squares be­low the four ovals us­ing dif­fer­ent shades of gray

>>> Solution (.py)

esOvalRect.png

Take also into ac­count that once a fill() or stroke() is set, it is used un­til fur­ther change. This script:

ovalsss.png

.py

newPage(100, 100)
fill(1)
stroke(0)
strokeWidth(2)
oval(10, 40, 50, 50)
oval(20, 30, 50, 50)
fill(.5)
oval(30, 20, 50, 50)
oval(40, 10, 50, 50)

is dif­fer­ent than this one:

ovalsss2.png

.py

newPage(100, 100)
stroke(0)
strokeWidth(2)
fill(1)
oval(10, 40, 50, 50)
fill(.75)
oval(20, 30, 50, 50)
fill(.5)
oval(30, 20, 50, 50)
fill(.25)
oval(40, 10, 50, 50)

Workbook

ex­er­cise 5.8

Your goal is to draw a black rec­tan­gle po­si­tioned in the mid­dle of the can­vas. Its height is equal to the height of the can­vas. The rec­tan­gle’s width in­stead changes ac­cord­ing to a vari­able “fac­tor” be­tween 0 and 1:

  • if 0, the rectangle’s width is equal to 0
  • if 1, the rectangle’s width is equal to the canvas width

>>> Solution (.py)

ex4.8.png

ex­er­cise 5.9

Draw two rec­tan­gles, one lean­ing on the left side of the can­vas, the other lean­ing on the right side. Their heights are equal to the can­vas. In­stead, each rec­tan­gle width is con­trolled by a vari­able “fac­tor” be­tween 0 and 1. When “fac­tor”:

  • is 0, you see no rectangle
  • is 1, you cannot see the white canvas background

>>> Solution (.py)

ex4.9.png

ex­er­cise 5.10

Draw four tri­an­gles. Fill each tri­an­gle with a dif­fer­ent shade of gray. Also, each tri­an­gle should have two ver­tices match­ing to con­tigu­ous can­vas cor­ners and the last ver­tex po­si­tioned in the mid­dle of the can­vas.

>>> Solution (.py)

ex4.10.png

ex­er­cise 5.11

Ex­tend the pre­vi­ous ex­er­cise: how could you make the height of each tri­an­gle re­act to a vari­able called “fac­tor”?

  • when “factor” is 0, you see no triangle (they have no height!)
  • when “factor” is 1, the vertices touch in the middle of the canvas

>>> Solution (.py)

ex4.11.png

ex­er­cise 5.12

Draw four cir­cles ver­ti­cally aligned in the mid­dle of a can­vas. The shapes should be equally spread out. A vari­able “ra­dius” con­trols the size of each cir­cle. The color of each shape should be cal­cu­lated ac­cord­ing to its hor­i­zon­tal po­si­tion in the can­vas: left darker, right lighter.

>>> Solution (.py)

ex4.12.png

ex­er­cise 5.13

Draw two crosses, a dark gray “plus” shape, and a light gray “mul­ti­ply” shape. The vari­able “fac­tor” con­trols the thick­ness of the crosses:

  • if 0, the thickness is 2pt
  • if 1, the thickness is 30pt

>>> Solution (.py)

ex4.13.png

ex­er­cise 5.14

Draw a grid with three columns and three rows. The pro­gram should al­low the pres­ence of a safe space be­tween the el­e­ments (of­ten re­ferred to as “gut­ter”). The “gut­ter” value should be ex­pressed in ty­po­graph­i­cal points.

>>> Solution (.py)

ex4.14.png